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. git操作合集

    目录 安装 下载 本地配置 创建用户凭证ssh 忽略文件 基础操作 新建仓库 克隆仓库 获取更新 推送更新 查看历史 版本回退 分支 别名 linux服务器 疑难问题 清除历史大文件 安装 下载 下载 ...

  2. laravel middleware

    当你使用larvel创建一个相对比较复杂的web网站时,往往你的routes文件就会变得很庞大.一般来说在开始网站编码之前,最好做一个整体规划,把这些route逻辑上划分为不同的group,每一个gr ...

  3. Windows 10 host where Credential Guard or Device Guard is enabled fails when running Workstation (2146361)

    To disable Device Guard or Credential Guard on Itanium based computers: Disable the group policy set ...

  4. Windows ->> Windows下安装MSI程序遇到2503和2502错误

    三个步骤可以解决这个问题: 1) 以管理员身份开启命令行模式并键入msiexec /package <msi文件路径> 2) 修改组策略 计算机配置 ->> 管理模板 -> ...

  5. select server

    server with select #include<stdio.h> #include<sys/types.h> #include<sys/socket.h> ...

  6. CentOS7路由、端口和服务排障常用命令

    1. ip route   显示主机基本路由信息 ~]# ip route default via 172.25.0.254 dev eth0  proto static  metric 1024 1 ...

  7. ubuntu搭建LAMP全教程及简单使用

    一:安装: 参考:http://jingyan.baidu.com/article/a681b0de36ad683b18434691.html 本经验向你展示如何在ubuntu14.04 环境下搭建a ...

  8. matplotlib点线 坐标刻度 3D图绘制(六)

    plot语句中支持除X,Y以外的参数,以字符串形式存在,来控制颜色.线型.点型等要素,语法形式为: plt.plot(X, Y, 'format', ...) 1 点和线的样式 颜色 参数color或 ...

  9. python的逻辑运算符

    函数就是特殊的变量(这个变量是不加引号的)  我们 如果直接打印一个函数的名字是直接打印这个函数的内存地址 python的运算符: 首先Python的运算符和其他语言的运算符都是一样的  都有:算数运 ...

  10. 将一种cell当做几种cell使用

    将一种cell当做几种cell使用 将一种cell当做几种cell用是有着一些意义的,比如,有时候,不同的cell之间差异很小,如果再派生一个cell出来,就会显得很麻烦,这时候,将这个cell当做几 ...