一. Servlet中的异步请求

在Servlet 3.0之前,Servlet采用Thread-Per-Request的方式处理请求,即每一次Http请求都由某一个线程从头到尾负责处理。如果要处理一些IO操作,以及访问数据库,调用第三方服务接口时,这种做法是十分耗时的。可以用代码测试一下:

同步方式处理: 

@WebServlet("/helloServlet")
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(Thread.currentThread()+"start...");
try {
sayHello();
} catch (InterruptedException e) {
e.printStackTrace();
}
resp.getWriter().write("hello...");
System.out.println(Thread.currentThread()+" end ...."); }
public void sayHello() throws InterruptedException {
Thread.sleep(3000);
}
}

以上代码,执行了一个3秒的方法,主线程在3秒方法执行完之后,才得到释放,这样处理效率非常不高。在Servlet 3.0引入了异步处理,然后在Servlet 3.1中又引入了非阻塞IO来进一步增强异步处理的性能。

异步方式处理业务逻辑:

@WebServlet(value = "/asyncServlet",asyncSupported = true)
public class AsyncHelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println(Thread.currentThread()+"主线程start..."+System.currentTimeMillis());
//1.支持异步处理 asyncSupported = true
//2.开启异步
AsyncContext asyncContext = req.startAsync();
//3.业务逻辑业务处理
asyncContext.start(new Runnable() {
@Override
public void run() {
try {
System.out.println(Thread.currentThread()+"副线程start..."+System.currentTimeMillis());
sayHello();
asyncContext.complete(); //4.获取响应
ServletResponse response = asyncContext.getResponse();
response.getWriter().write("hello,async...");
System.out.println(Thread.currentThread()+"副线程end..."+System.currentTimeMillis());
} catch (InterruptedException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
});
System.out.println(Thread.currentThread()+"主线程end..."+System.currentTimeMillis()); //测试结果,主线程接受请求处理,并立即得到释放,副线程处理业务逻辑
/**
* Thread[http-apr-8080-exec-9,5,main]主线程start...1545911791802
* Thread[http-apr-8080-exec-9,5,main]主线程end...1545911791806
* Thread[http-apr-8080-exec-10,5,main]副线程start...1545911791806
* Thread[http-apr-8080-exec-10,5,main]副线程end...1545911794807
*/
}
public void sayHello() throws InterruptedException {
Thread.sleep(3000);
}

上面的代码中,主线程在Servlet被执行到时,立刻得到了释放,然后在副线程中调用了sayHello方法,3秒后副线程得到释放,这就是异步Servlet最简单的一个demo演示。

. springmvc中的异步请求

1.返回Callable<>接口

1)Spring异步处理,将Callable提交到TaskExecutor 使用一个隔离的线程进行执行。

2)DispacherServlet和所有的Filter退出线程,但是response保持打开状态

3)Callbale返回结果,springMVC将请求重新派发给容器,恢复之前的处理

4)根据Callable返回的结果,springMVC继续进行视图渲染流程等(从请求-视图渲染)。

代码如下:

  @RequestMapping("/callable")
@ResponseBody
public Callable<String> async01(){
System.out.println("主线程"+Thread.currentThread()+"Start======>"+System.currentTimeMillis());
Callable<String> callable = new Callable<String>() { @Override
public String call() throws Exception {
System.out.println("副线程"+Thread.currentThread()+"Start======>"+System.currentTimeMillis());
Thread.sleep(3000);
System.out.println("副线程"+Thread.currentThread()+"End======>"+System.currentTimeMillis());
return "Callable<String> async01()";
}
};
System.out.println("主线程"+Thread.currentThread()+"End======>"+System.currentTimeMillis()); return callable;
}

测试运行结果:

      MyInterceptor preHandle..
拦截器拦截的请求:/callable
主线程Thread[http-apr-8080-exec-9,5,main]Start======>1545965335805
主线程Thread[http-apr-8080-exec-9,5,main]End======>1545965335805
==============DispacherServlet及所有的Filter退出线程=============
========================等待Callable执行========================
副线程Thread[MvcAsync1,5,main]Start======>1545965335811
副线程Thread[MvcAsync1,5,main]End======>1545965338811
========================Callable执行完成========================
MyInterceptor preHandle..
拦截器拦截的请求:/callable
MyInterceptor postHandle..(Callable的之前的返回值就是目标方法的返回值)
MyInterceptor afterCompletion..

2.使用DeferredResult<>实现异步

public  class DeferredResultQueue {
public static Queue<DeferredResult<Object>> queue=new ConcurrentLinkedDeque<>();
public static void save(DeferredResult<Object> deferredResult){
queue.add(deferredResult);
}
public static DeferredResult<Object> get(){
return queue.poll();
}
}
@Controller
public class MyDeferredResultController{
/**
* 一个线程创建订单,一个线程等待处理订单
* @return
*/
@ResponseBody
@RequestMapping("/createOrder")
public DeferredResult<Object> createOrder(){
//设置等待时间,3秒后等待超时
DeferredResult<Object> deferredResult = new DeferredResult<>((long)3000,"createFail");
DeferredResultQueue.save(deferredResult);
return deferredResult;
}
@ResponseBody
@RequestMapping("/create")
public String create(){
String order = UUID.randomUUID().toString();
DeferredResult<Object> deferredResult = DeferredResultQueue.get();
deferredResult.setResult(order);
return "success===>"+order;
}
}

上述代码模拟了创建订单的过程,运行时,先访问/createOrder请求,该请求会等待3秒(3秒期间内没有订单创建会超时),然后访问另一个请求/create,该请求创建号订单后,/createOrder立即接收到了刚创建好的订单,两者是异步执行的。

Spring注解开发系列Ⅸ --- 异步请求的更多相关文章

  1. Spring注解开发系列Ⅵ --- AOP&事务

    注解开发 --- AOP AOP称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,横向重复,纵向抽取.详细的AO ...

  2. Spring注解开发系列专栏

    这个系列主要是讲Spring注解的使用,可以为后面SpringBoot的学习带来一定的帮助.我觉得从Spring直接过度到SpringBoot还是有点快,还是得需要一个演变的过程.从Spring开发, ...

  3. Spring注解开发系列VIII --- SpringMVC

    SpringMVC是三层架构中的控制层部分,有过JavaWEB开发经验的同学一定很熟悉它的使用了.这边有我之前整理的SpringMVC相关的链接: 1.SpringMVC入门 2.SpringMVC进 ...

  4. Spring注解开发系列VII --- Servlet3.0

    Servlet3.0简介 Servlet 3.0 作为 Java EE 6 规范体系中一员,随着 Java EE 6 规范一起发布.该版本在前一版本(Servlet 2.5)的基础上提供了若干新特性用 ...

  5. Spring注解开发系列Ⅰ--- 组件注册(上)

    传统的Spring做法是使用.xml文件来对bean进行注入或者是配置aop.事物,这么做有两个缺点:1.如果所有的内容都配置在.xml文件中,那么.xml文件将会十分庞大:如果按需求分开.xml文件 ...

  6. Spring注解开发系列Ⅴ --- 自动装配&Profile

    自动装配: spring利用依赖注入和DI完成对IOC容器中各个组件的依赖关系赋值.自动装配的优点有: 自动装配可以大大地减少属性和构造器参数的指派. 自动装配也可以在解析对象时更新配置. 自动装配的 ...

  7. Spring注解开发系列Ⅱ --- 组件注册(下)

    1.@Import注册组件 @Import主要功能是通过导入的方式实现把实例加入springIOC容器中, /** * 给容器注册组件 * 1.包扫描+组件标注注解(@Controller,@Serv ...

  8. Spring注解开发系列Ⅲ --- 生命周期

    Bean的生命周期 Spring Bean 的生命周期在整个 Spring 中占有很重要的位置,掌握这些可以加深对 Spring 的理解. 首先看下生命周期图: 再谈生命周期之前有一点需要先明确: S ...

  9. Spring注解开发系列Ⅳ --- 属性赋值

    在Spring框架中,属性的注入我们有多种方式,我们可以通过构造方法注入,可以通过set方法注入,也可以通过p名称空间注入,方式多种多样,对于复杂的数据类型比如对象.数组.List集合.map集合.P ...

随机推荐

  1. callback、promise和async、await的使用方法

    callback 回调是一个函数被作为一个参数传递到另一个函数里,在那个函数执行完后再执行.通俗的讲就是 B函数被作为参数传递到A函数里,在A函数执行完后再执行B. promise Promise 是 ...

  2. Java正则表达式学习与记录

    转载自:http://www.runoob.com/java/java-regular-expressions.html 正则表达式定义了字符串的模式,用于搜索.编辑或处理文本. 1.正则表达式中字符 ...

  3. 【阿里云IoT+YF3300】10.快速开发188协议设备驱动

    188协议的全称为CJ-T188-2004 <户用计量仪表数据传输技术条件>,是针对水表.燃气表.热量表和其他集中采集的一个国家行业标准协议. YFIOs就是YFSoft I/O Serv ...

  4. 洛谷$1220$ 关路灯 记搜/$DP$

    \(Sol\) 约定\(pos\)为老张所处的位置的路灯号,\(i<pos,j>pos\). 显然,如果\(i\)和\(j\)都关了,那么它们之间的所有灯一定也都关了. 设\(f[i][j ...

  5. Windows Live Writer 语法高亮

    1.WindowsLiveWriter.CNBlogs.CodeHighlighter.rar 这个插件生成的高亮代码与网页上的一模一样,插入后即可立即显示效果,不过貌似它必须联网才能实时显示效果,因 ...

  6. php strcmp函数漏洞

    strcmp函数漏洞 适用5.3版本以前的php 函数作用:字符串比较 要求传入字符串.如果传入非字符串呢? 结果函数报错!但是函数返回“0”  . 虽然报错了但函数的判断却是“相等” 如何传入非字符 ...

  7. 浅谈Java三大特性之封装

    目录 前言 好处 介绍(实现方式) 示例 小结 感谢阅读!!! 三大特性之浅谈封装 前言 英语:Encapsulation--封装,包装. 面向对象的封装与真实世界的目的是一样的.封装能够使外部访问者 ...

  8. 计算n的阶乘

    题目描述 定义一个函数,传入一个整数n,打印n!的值比如:传入3打印:6  <====1*2*3 输入 整数n    输出 整数n的阶乘 样例输入 Copy 3 样例输出 Copy 6 x=in ...

  9. rust 打印当前时间

    let now = time::now();let f_now = time::strftime("%Y-%m-%dT%H:%M:%S", &now).unwrap();p ...

  10. Spring Boot从零入门3_创建Hello World及项目剖析

    目录 1 前言 2 名词术语 3 创建Hello World项目 3.1 基于STS4创建项目 3.2 使用Spring Initializr Website创建项目并导入 3.3 基于Spring ...