Spring注解开发系列Ⅸ --- 异步请求
一. 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注解开发系列Ⅸ --- 异步请求的更多相关文章
- Spring注解开发系列Ⅵ --- AOP&事务
注解开发 --- AOP AOP称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,横向重复,纵向抽取.详细的AO ...
- Spring注解开发系列专栏
这个系列主要是讲Spring注解的使用,可以为后面SpringBoot的学习带来一定的帮助.我觉得从Spring直接过度到SpringBoot还是有点快,还是得需要一个演变的过程.从Spring开发, ...
- Spring注解开发系列VIII --- SpringMVC
SpringMVC是三层架构中的控制层部分,有过JavaWEB开发经验的同学一定很熟悉它的使用了.这边有我之前整理的SpringMVC相关的链接: 1.SpringMVC入门 2.SpringMVC进 ...
- Spring注解开发系列VII --- Servlet3.0
Servlet3.0简介 Servlet 3.0 作为 Java EE 6 规范体系中一员,随着 Java EE 6 规范一起发布.该版本在前一版本(Servlet 2.5)的基础上提供了若干新特性用 ...
- Spring注解开发系列Ⅰ--- 组件注册(上)
传统的Spring做法是使用.xml文件来对bean进行注入或者是配置aop.事物,这么做有两个缺点:1.如果所有的内容都配置在.xml文件中,那么.xml文件将会十分庞大:如果按需求分开.xml文件 ...
- Spring注解开发系列Ⅴ --- 自动装配&Profile
自动装配: spring利用依赖注入和DI完成对IOC容器中各个组件的依赖关系赋值.自动装配的优点有: 自动装配可以大大地减少属性和构造器参数的指派. 自动装配也可以在解析对象时更新配置. 自动装配的 ...
- Spring注解开发系列Ⅱ --- 组件注册(下)
1.@Import注册组件 @Import主要功能是通过导入的方式实现把实例加入springIOC容器中, /** * 给容器注册组件 * 1.包扫描+组件标注注解(@Controller,@Serv ...
- Spring注解开发系列Ⅲ --- 生命周期
Bean的生命周期 Spring Bean 的生命周期在整个 Spring 中占有很重要的位置,掌握这些可以加深对 Spring 的理解. 首先看下生命周期图: 再谈生命周期之前有一点需要先明确: S ...
- Spring注解开发系列Ⅳ --- 属性赋值
在Spring框架中,属性的注入我们有多种方式,我们可以通过构造方法注入,可以通过set方法注入,也可以通过p名称空间注入,方式多种多样,对于复杂的数据类型比如对象.数组.List集合.map集合.P ...
随机推荐
- centos6.x将python2.6升级到2.7
一,安装开发工具和Python2.7(1)查看当前python版本 python -V Python 2.6.6 (2)下载Python-2.7.3 wget http://python.org/ft ...
- 20191024-3 互评Alpha阶段作品——构建之法组
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2019fall/homework/9860 基于NABCD评论作品,及改进建议 1.根据(不限于)NABCD评 ...
- 洛谷$P3645\ [APIO2015]$雅加达的摩天楼 最短路
正解:最短路 解题报告: 传送门$QwQ$ 考虑暴力连边,发现最多有$n^2$条边.于是考虑分块 对于长度$p_i$小于等于$\sqrt(n)$的边,建立子图$d=p_i$.说下关于子图$d$的定义? ...
- shell脚本配置maven
#!/bin/bash # maven install mvnpath=/usr/local/maven # 不存在 if [ ! -d "$mvnpath" ]; then ec ...
- Elasticsearch 节点磁盘使用率过高,导致ES集群索引无副本
目录 一.问题 二.问题的原因 三.问题解决的办法 1. 扩大磁盘 2. 删除部分历史索引 3. 更改es设置 四.扩展 一.问题 最近在查看线上的 es,发现最近2天的索引没有副本,集群的状态也是为 ...
- linux dubbo-admin-2.6.0 环境搭建
1.去maven官网下载apache-maven-3.6.2-bin.tar.gz安装包 2.上传至linux服务器中 3.解压maven安装包 tar -zxvf apache-maven-3.6. ...
- Q: 字符串的修改
题目描述 怎么样,前面的题还可以吧~ 依旧是字符串处理,设A和B是两个字符串.我们要用最少的字符操作次数,将字符串A转换为字符串B.这里所说的字符操作共有三种: 1. 删除一个字符: 2. 插入一个字 ...
- Codeforces Round #612 (Div. 2) 前四题题解
这场比赛的出题人挺有意思,全部magic成了青色. 还有题目中的图片特别有趣. 晚上没打,开virtual contest打的,就会前三道,我太菜了. 最后看着题解补了第四道. 比赛传送门 A. An ...
- 用实例理解设计模式——代理模式(Python版)
代理模式:为其他对象提供一种代理以控制对这个对象的访问. 在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 代理模式分为: 静态代理 动态代 ...
- 4.eclipse中导入别人用的源代码问题
最近在导入别人用的源代码问题时,出现两个问题: 问题一:提示无法解析导入,如下图: 解决方法:删除项目下的module-info.java文件即可,或者在创建项目时将创建module-info.jav ...