Spring MVC 对于异步请求处理的两种方式

场景: Tomcat对于主线程性能瓶颈,当Tomcat请求并发数过多时,当线程数满时,就会出现请求等待Tomcat处理,这个时候可以使用子线程处理业务逻辑,主线程只是处理返回请求,这样可以大大提高Tomcat的吞吐量。

1. Callable

1. 使用Callable返回异步信息

- 对于前台用户来说,只是一个同步的请求,根本感觉不到后台的异步处理。

- 后台直接返回Callable,是由Tomcat返回给前台, 而Callable数据会等待返回结果给前台。

- 代码

    //简单的Mapping映射请求
@GetMapping("order")
public Callable<String> order1 () {
_LOGGER.info("主线程开始");
Callable<String> callable = () -> {
_LOGGER.info("副线程启动");
//模拟业务逻辑-需要一秒来处理
TimeUnit.SECONDS.sleep(1);
_LOGGER.info("副线程返回");
return "success";
};
_LOGGER.info("主线程返回");
return callable;
}
//日志输出,可以看到主线程直接返回, 而子线程等待一秒后,这时候前台返回数据
//2018-04-11 11:27:54.584 INFO 1912 --- [nio-8010-exec-8] org.ko.web.async.AsyncController : 主线程开始
//2018-04-11 11:27:54.584 INFO 1912 --- [nio-8010-exec-8] org.ko.web.async.AsyncController : 主线程返回
//2018-04-11 11:27:54.592 INFO 1912 --- [ MvcAsync1] org.ko.web.async.AsyncController : 副线程启动
//2018-04-11 11:27:55.592 INFO 1912 --- [ MvcAsync1] org.ko.web.async.AsyncController : 副线程返回
- 使用Callable 对于前台来说是和正常请求一样的,对于后台来说却可以大大增加Tomcat吞吐量。

当然有些场景, Callable并不能解决,比如说:我们访问A接口,A接口调用三方的服务,服务回调B接口,这种情况就没办法使用Callable了,这个时候可以使用DeferredResult

2. DeferredResult

2. 使用DeferredResult异步处理复杂场景,线程间数据传递

- DeferredResult: 对于用户来说,只是一个同步请求,而后台是分开两个接口。

- 同Callable一样,直接返回给前台,后续再像DeferredResult中放入值,前台直接获取数据。

- 代码

    @GetMapping("order")
public DeferredResult<String> order () throws InterruptedException {
_LOGGER.info("主线程开始");
String orderNumber = RandomStringUtils.randomNumeric(8);
mockQueue.setPlaceOrder(orderNumber);
DeferredResult<String> result = new DeferredResult<>();
deferredResultHolder.getMap().put(orderNumber, result);
return result;
}
- PlaceOrder中创建线程模拟双接口
public void setPlaceOrder(String placeOrder) throws InterruptedException {
_LOGGER.info("接到下单请求");
new Thread(() -> {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.completeOrder = placeOrder;
_LOGGER.info("下单请求处理完毕");
}).start();
}
- 实现ApplicationListener模拟回调,最后返回给前台
    @Component
public class QueueListener implements ApplicationListener<ContextRefreshedEvent>{ private static final Logger _LOGGER = LoggerFactory.getLogger(QueueListener.class); @Autowired
private MockQueue mockQueue; @Autowired DeferredResultHolder deferredResultHolder; @Override
public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
new Thread(() -> {
while (true) {
if (StringUtils.isNotBlank(mockQueue.getCompleteOrder())) {
String orderNumber = mockQueue.getCompleteOrder();
_LOGGER.info("返回订单处理结果: {}", orderNumber);
deferredResultHolder.getMap().get(orderNumber).setResult("place order success;");
mockQueue.setCompleteOrder(null);
} else {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
}
}
- Holder实现
    @Component
public class DeferredResultHolder {
/**
* 订单处理结果
*/
private Map<String, DeferredResult<String>> map = new HashMap<>();
public Map<String, DeferredResult<String>> getMap() {
return map;
} public void setMap(Map<String, DeferredResult<String>> map) {
this.map = map;
}
}
- DeferredResult比较适合一些比较复杂的业务场景,提升性能。
- 有个问题,当使用分布式部署,调用链走的不是同一个实例时,DeferredResult的处理有可能会出现问题。

3. 异步调优

  • 对异步处理调优的一些参数配置,Spring默认异步线程是不使用线程池的,可以自己设定一些可以重用的线程

  • 继承WebMvcConfigurerAdapter重写configureAsyncSupport()方法

    //@Configuration
public class WebConfig extends WebMvcConfigurerAdapter { @Override
public void configureAsyncSupport(AsyncSupportConfigurer configurer) {
//注册callable拦截器
configurer.registerCallableInterceptors();
//注册deferredResult拦截器
configurer.registerDeferredResultInterceptors()
//异步请求超时时间
configurer.setDefaultTimeout()
//设定异步请求线程池callable等, spring默认线程不可重用
configurer.setTaskExecutor()
}
}

4. 代码

Spring Web Async异步处理#Callable #DeferredResult的更多相关文章

  1. Spring Boot @Async 异步任务执行

    1.任务执行和调度 Spring用TaskExecutor和TaskScheduler接口提供了异步执行和调度任务的抽象. Spring的TaskExecutor和java.util.concurre ...

  2. 利用Spring的@Async异步处理改善web应用中耗时操作的用户体验

    Web应用中,有时会遇到一些耗时很长的操作(比如:在后台生成100张报表再呈现,或 从ftp下载若干文件,综合处理后再返回给页面下载),用户在网页上点完按钮后,通常会遇到二个问题:页面超时.看不到处理 ...

  3. spring boot @Async异步注解上下文透传

    上一篇文章说到,之前使用了@Async注解,子线程无法获取到上下文信息,导致流量无法打到灰度,然后改成 线程池的方式,每次调用异步调用的时候都手动透传 上下文(硬编码)解决了问题. 后面查阅了资料,找 ...

  4. 使用spring的@Async异步执行方法

    应用场景: 1.某些耗时较长的而用户不需要等待该方法的处理结果 2.某些耗时较长的方法,后面的程序不需要用到这个方法的处理结果时 在spring的配置文件中加入对异步执行的支持 <beans x ...

  5. Spring Boot Async异步执行

    异步调用就是不用等待结果的返回就执行后面的逻辑,同步调用则需要等带结果再执行后面的逻辑. 通常我们使用异步操作都会去创建一个线程执行一段逻辑,然后把这个线程丢到线程池中去执行,代码如下: Execut ...

  6. spring的@Async异步使用

    pring的@Async功能,用的时候一定要注意: 1.异步方法和调用类不要在同一个类中. 2.xml里需要加入这一行 <task:annotation-driven/> 下面的可以直接粘 ...

  7. SpringMVC异步调用,Callable和DeferredResult的使用

    Callable和DeferredResult都是springMVC里面的异步调用,Callable主要用来处理一些简单的逻辑,DeferredResult主要用于处理一些复杂逻辑 1.Callabl ...

  8. spring boot(17)-@Async异步

    验证码的异步机制 上一篇讲过可以用邮件发验证码,通常我们在某网站发验证码时,首先会提示验证码已发送,请检查邮箱或者短信,这就是图中的1和3.然而此时查看邮箱或短信可能并没有收到验证码,往往要过几秒种才 ...

  9. Spring Boot (18) @Async异步

    通常我们在某网站发送邮件验证码时,首先会提示验证码已发送,然而此时可能没有收到验证码,过几秒种才真正的收到.如果是同步会先验证发送是否成功然后再通知,如果是异步可以先通知用户已发送,并释放请求,然后再 ...

随机推荐

  1. CentOS安装和部署SVN服务器

    1.安装SVN 通过yum安装svn [root@localhost webber]# yum install subversion 查看svn是否安装成功 [root@localhost webbe ...

  2. JMeter中文版用户手册

    1.1 简介 使用JMeter通常会有以下步骤: 1.1.1 创建测试计划 首先,运行JMeter图形化界面. 然后在文件菜单中选择Templates…->Recording,通过浏览器录制We ...

  3. 初始python(三)

    1. 循环 if, while, forbreak : 结束整个循环continue :跳出当前这次循环,但不结束整个循环else :结束整个循环后才执行,不能与break合用,但可以与continu ...

  4. [EffectiveC++]item02:尽量以const,enum,inline代替#define

  5. boost编译

    从boost官网( http://www.boost.org )上下载最新的boost版本,现在最新是1.64版本,解压到自定义目录(我解压到了E盘E:\boost_1_64_0\,最终的目录结构是E ...

  6. [转]solr系统query检索词特殊字符的处理

    原文地址:http://blog.csdn.net/wgw335363240/article/details/39889979 solr是基于 lucence开发的应用,如果query中带有非法字符串 ...

  7. Python数据类型-字典

    字典(dict) 字典是key:value形式的一种表达形式,例如在Java中有map,JavaScript中的json,Redis中的hash等等这些形式.字典可以存储任意的对象,也可以是不同的数据 ...

  8. java中JVM的原理重温【转】

    一.基础理论知识 1.java虚拟机的生命周期: Java虚拟机的生命周期 一个运行中的Java虚拟机有着一个清晰的任务:执行Java程序.程序开始执行时他才运行,程序结束时他就停止.你在同一台机器上 ...

  9. Tomcat中容器是什么以及容器与容器之间的数量关系。

    Tomcat容器到底是什么 学java有一小段时间了,一直使用Tomcat,也知道Tomcat是一个大的Servlet容器,里面还有许多子容器,容器之间都是相互嵌套的.也看过一下接收Tomcat的文章 ...

  10. curl 抓取页面信息

    <?php // 1. 初始化 $ch = curl_init(); // 2. 设置选项,包括URL curl_setopt($ch, CURLOPT_URL, "www" ...