SpringBoot实践之---使用异步请求,提高系统的吞吐量

原文地址: spring boot--使用异步请求,提高系统的吞吐量

前言:

在我们的实际生产中,常常会遇到下面的这种情况,某个请求非常耗时(大约5s返回),当大量的访问该请求的时候,再请求其他服务时,会造成没有连接使用的情况,造成这种现象的主要原因是,我们的容器(tomcat)中线程的数量是一定的,例如500个,当这500个线程都用来请求服务的时候,再有请求进来,就没有多余的连接可用了,只能拒绝连接。要是我们在请求耗时服务的时候,能够异步请求(请求到controller中时,则容器线程直接返回,然后使用系统内部的线程来执行耗时的服务,等到服务有返回的时候,再将请求返回给客户端),那么系统的吞吐量就会得到很大程度的提升了。当然,大家可以直接使用Hystrix的资源隔离来实现,今天我们的重点是spring mvc是怎么来实现这种异步请求的。

一、使用Callable来实现

controller如下:

[java]  view plain  copy
  1. @RestController  
  2. public class HelloController {  
  3.   
  4.     private static final Logger logger = LoggerFactory.getLogger(HelloController.class);  
  5.       
  6.     @Autowired  
  7.     private HelloService hello;  
  8.   
  9.     @GetMapping("/helloworld")  
  10.     public String helloWorldController() {  
  11.         return hello.sayHello();  
  12.     }  
  13.   
  14.     /** 
  15.      * 异步调用restful 
  16.      * 当controller返回值是Callable的时候,springmvc就会启动一个线程将Callable交给TaskExecutor去处理 
  17.      * 然后DispatcherServlet还有所有的spring拦截器都退出主线程,然后把response保持打开的状态 
  18.      * 当Callable执行结束之后,springmvc就会重新启动分配一个request请求,然后DispatcherServlet就重新 
  19.      * 调用和处理Callable异步执行的返回结果, 然后返回视图 
  20.      *  
  21.      * @return 
  22.      */  
  23.     @GetMapping("/hello")  
  24.     public Callable<String> helloController() {  
  25.         logger.info(Thread.currentThread().getName() + " 进入helloController方法");  
  26.         Callable<String> callable = new Callable<String>() {  
  27.   
  28.             @Override  
  29.             public String call() throws Exception {  
  30.                 logger.info(Thread.currentThread().getName() + " 进入call方法");  
  31.                 String say = hello.sayHello();  
  32.                 logger.info(Thread.currentThread().getName() + " 从helloService方法返回");  
  33.                 return say;  
  34.             }  
  35.         };  
  36.         logger.info(Thread.currentThread().getName() + " 从helloController方法返回");  
  37.         return callable;  
  38.     }  
  39. }  

我们首先来看下上面这两个请求的区别

下面这个是没有使用异步请求的

[java]  view plain  copy
  1. 2017-12-07 18:05:42.351  INFO 3020 --- [nio-8060-exec-5] c.travelsky.controller.HelloController   : http-nio-8060-exec-5 进入helloWorldController方法  
  2. 2017-12-07 18:05:42.351  INFO 3020 --- [nio-8060-exec-5] com.travelsky.service.HelloService       : http-nio-8060-exec-5 进入sayHello方法!  
  3. 2017-12-07 18:05:44.351  INFO 3020 --- [nio-8060-exec-5] c.travelsky.controller.HelloController   : http-nio-8060-exec-5 从helloWorldController方法返回  

我们可以看到,请求从头到尾都只有一个线程,并且整个请求耗费了2s钟的时间。

下面,我们再来看下使用Callable异步请求的结果:

[java]  view plain  copy
  1. 2017-12-07 18:11:55.671  INFO 6196 --- [nio-8060-exec-1] c.travelsky.controller.HelloController   : http-nio-8060-exec-1 进入helloController方法  
  2. 2017-12-07 18:11:55.672  INFO 6196 --- [nio-8060-exec-1] c.travelsky.controller.HelloController   : http-nio-8060-exec-1 从helloController方法返回  
  3. 2017-12-07 18:11:55.676  INFO 6196 --- [nio-8060-exec-1] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-1 进入afterConcurrentHandlingStarted方法  
  4. 2017-12-07 18:11:55.676  INFO 6196 --- [      MvcAsync1] c.travelsky.controller.HelloController   : MvcAsync1 进入call方法  
  5. 2017-12-07 18:11:55.676  INFO 6196 --- [      MvcAsync1] com.travelsky.service.HelloService       : MvcAsync1 进入sayHello方法!  
  6. 2017-12-07 18:11:57.677  INFO 6196 --- [      MvcAsync1] c.travelsky.controller.HelloController   : MvcAsync1 从helloService方法返回  
  7. 2017-12-07 18:11:57.721  INFO 6196 --- [nio-8060-exec-2] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-2服务调用完成,返回结果给客户端  

从上面的结果中,我们可以看出,容器的线程http-nio-8060-exec-1这个线程进入controller之后,就立即返回了,具体的服务调用是通过MvcAsync2这个线程来做的,当服务执行完要返回后,容器会再启一个新的线程http-nio-8060-exec-2来将结果返回给客户端或浏览器,整个过程response都是打开的,当有返回的时候,再从server端推到response中去。


1、异步调用的另一种方式 WebAsyncTask

上面的示例是通过callable来实现的异步调用,其实还可以通过WebAsyncTask,也能实现异步调用,下面看示例:

[java]  view plain  copy
  1. @RestController  
  2. public class HelloController {  
  3.   
  4.     private static final Logger logger = LoggerFactory.getLogger(HelloController.class);  
  5.       
  6.     @Autowired  
  7.     private HelloService hello;  
  8.   
  9.         /** 
  10.      * 带超时时间的异步请求 通过WebAsyncTask自定义客户端超时间 
  11.      *  
  12.      * @return 
  13.      */  
  14.     @GetMapping("/world")  
  15.     public WebAsyncTask<String> worldController() {  
  16.         logger.info(Thread.currentThread().getName() + " 进入helloController方法");  
  17.   
  18.         // 3s钟没返回,则认为超时  
  19.         WebAsyncTask<String> webAsyncTask = new WebAsyncTask<>(3000new Callable<String>() {  
  20.   
  21.             @Override  
  22.             public String call() throws Exception {  
  23.                 logger.info(Thread.currentThread().getName() + " 进入call方法");  
  24.                 String say = hello.sayHello();  
  25.                 logger.info(Thread.currentThread().getName() + " 从helloService方法返回");  
  26.                 return say;  
  27.             }  
  28.         });  
  29.         logger.info(Thread.currentThread().getName() + " 从helloController方法返回");  
  30.   
  31.         webAsyncTask.onCompletion(new Runnable() {  
  32.   
  33.             @Override  
  34.             public void run() {  
  35.                 logger.info(Thread.currentThread().getName() + " 执行完毕");  
  36.             }  
  37.         });  
  38.   
  39.         webAsyncTask.onTimeout(new Callable<String>() {  
  40.   
  41.             @Override  
  42.             public String call() throws Exception {  
  43.                 logger.info(Thread.currentThread().getName() + " onTimeout");  
  44.                 // 超时的时候,直接抛异常,让外层统一处理超时异常  
  45.                 throw new TimeoutException("调用超时");  
  46.             }  
  47.         });  
  48.         return webAsyncTask;  
  49.     }  
  50.   
  51.     /** 
  52.      * 异步调用,异常处理,详细的处理流程见MyExceptionHandler类 
  53.      *  
  54.      * @return 
  55.      */  
  56.     @GetMapping("/exception")  
  57.     public WebAsyncTask<String> exceptionController() {  
  58.         logger.info(Thread.currentThread().getName() + " 进入helloController方法");  
  59.         Callable<String> callable = new Callable<String>() {  
  60.   
  61.             @Override  
  62.             public String call() throws Exception {  
  63.                 logger.info(Thread.currentThread().getName() + " 进入call方法");  
  64.                 throw new TimeoutException("调用超时!");  
  65.             }  
  66.         };  
  67.         logger.info(Thread.currentThread().getName() + " 从helloController方法返回");  
  68.         return new WebAsyncTask<>(20000, callable);  
  69.     }  
  70.   
  71. }  

运行结果如下:

[java]  view plain  copy
  1. 2017-12-07 19:10:26.582  INFO 6196 --- [nio-8060-exec-4] c.travelsky.controller.HelloController   : http-nio-8060-exec-4 进入helloController方法  
  2. 2017-12-07 19:10:26.585  INFO 6196 --- [nio-8060-exec-4] c.travelsky.controller.HelloController   : http-nio-8060-exec-4 从helloController方法返回  
  3. 2017-12-07 19:10:26.589  INFO 6196 --- [nio-8060-exec-4] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-4 进入afterConcurrentHandlingStarted方法  
  4. 2017-12-07 19:10:26.591  INFO 6196 --- [      MvcAsync2] c.travelsky.controller.HelloController   : MvcAsync2 进入call方法  
  5. 2017-12-07 19:10:26.591  INFO 6196 --- [      MvcAsync2] com.travelsky.service.HelloService       : MvcAsync2 进入sayHello方法!  
  6. 2017-12-07 19:10:28.591  INFO 6196 --- [      MvcAsync2] c.travelsky.controller.HelloController   : MvcAsync2 从helloService方法返回  
  7. 2017-12-07 19:10:28.600  INFO 6196 --- [nio-8060-exec-5] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-5服务调用完成,返回结果给客户端  
  8. 2017-12-07 19:10:28.601  INFO 6196 --- [nio-8060-exec-5] c.travelsky.controller.HelloController   : http-nio-8060-exec-5 执行完毕  

这种方式和上面的callable方式最大的区别就是,WebAsyncTask支持超时,并且还提供了两个回调函数,分别是onCompletion和onTimeout,顾名思义,这两个回调函数分别在执行完成和超时的时候回调。


3、Deferred方式实现异步调用

在我们是生产中,往往会遇到这样的情景,controller中调用的方法很多都是和第三方有关的,例如JMS,定时任务,队列等,拿JMS来说,比如controller里面的服务需要从JMS中拿到返回值,才能给客户端返回,而从JMS拿值这个过程也是异步的,这个时候,我们就可以通过Deferred来实现整个的异步调用。

首先,我们来模拟一个长时间调用的任务,代码如下:

[java]  view plain  copy
  1. @Component  
  2. public class LongTimeTask {  
  3.     private final Logger logger = LoggerFactory.getLogger(this.getClass());  
  4.     @Async  
  5.     public void execute(DeferredResult<String> deferred){  
  6.         logger.info(Thread.currentThread().getName() + "进入 taskService 的 execute方法");  
  7.         try {  
  8.             // 模拟长时间任务调用,睡眠2s  
  9.             TimeUnit.SECONDS.sleep(2);  
  10.             // 2s后给Deferred发送成功消息,告诉Deferred,我这边已经处理完了,可以返回给客户端了  
  11.             deferred.setResult("world");  
  12.         } catch (InterruptedException e) {  
  13.             e.printStackTrace();  
  14.         }  
  15.     }  
  16. }  

接着,我们就来实现异步调用,controller如下:

[java]  view plain  copy
  1. @RestController  
  2. public class AsyncDeferredController {  
  3.     private final Logger logger = LoggerFactory.getLogger(this.getClass());  
  4.     private final LongTimeTask taskService;  
  5.       
  6.     @Autowired  
  7.     public AsyncDeferredController(LongTimeTask taskService) {  
  8.         this.taskService = taskService;  
  9.     }  
  10.       
  11.     @GetMapping("/deferred")  
  12.     public DeferredResult<String> executeSlowTask() {  
  13.         logger.info(Thread.currentThread().getName() + "进入executeSlowTask方法");  
  14.         DeferredResult<String> deferredResult = new DeferredResult<>();  
  15.         // 调用长时间执行任务  
  16.         taskService.execute(deferredResult);  
  17.         // 当长时间任务中使用deferred.setResult("world");这个方法时,会从长时间任务中返回,继续controller里面的流程  
  18.         logger.info(Thread.currentThread().getName() + "从executeSlowTask方法返回");  
  19.         // 超时的回调方法  
  20.         deferredResult.onTimeout(new Runnable(){  
  21.           
  22.             @Override  
  23.             public void run() {  
  24.                 logger.info(Thread.currentThread().getName() + " onTimeout");  
  25.                 // 返回超时信息  
  26.                 deferredResult.setErrorResult("time out!");  
  27.             }  
  28.         });  
  29.           
  30.         // 处理完成的回调方法,无论是超时还是处理成功,都会进入这个回调方法  
  31.         deferredResult.onCompletion(new Runnable(){  
  32.           
  33.             @Override  
  34.             public void run() {  
  35.                 logger.info(Thread.currentThread().getName() + " onCompletion");  
  36.             }  
  37.         });  
  38.           
  39.         return deferredResult;  
  40.     }  
  41. }  

执行结果如下:

[java]  view plain  copy
  1. 2017-12-07 19:25:40.192  INFO 6196 --- [nio-8060-exec-7] c.t.controller.AsyncDeferredController   : http-nio-8060-exec-7进入executeSlowTask方法  
  2. 2017-12-07 19:25:40.193  INFO 6196 --- [nio-8060-exec-7] .s.a.AnnotationAsyncExecutionInterceptor : No TaskExecutor bean found for async processing  
  3. 2017-12-07 19:25:40.194  INFO 6196 --- [nio-8060-exec-7] c.t.controller.AsyncDeferredController   : http-nio-8060-exec-7从executeSlowTask方法返回  
  4. 2017-12-07 19:25:40.198  INFO 6196 --- [nio-8060-exec-7] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-7 进入afterConcurrentHandlingStarted方法  
  5. 2017-12-07 19:25:40.202  INFO 6196 --- [cTaskExecutor-1] com.travelsky.controller.LongTimeTask    : SimpleAsyncTaskExecutor-1进入 taskService 的 execute方法  
  6. 2017-12-07 19:25:42.212  INFO 6196 --- [nio-8060-exec-8] c.t.i.MyAsyncHandlerInterceptor          : http-nio-8060-exec-8服务调用完成,返回结果给客户端  
  7. 2017-12-07 19:25:42.213  INFO 6196 --- [nio-8060-exec-8] c.t.controller.AsyncDeferredController   : http-nio-8060-exec-8 onCompletion  

从上面的执行结果不难看出,容器线程会立刻返回,应用程序使用线程池里面的cTaskExecutor-1线程来完成长时间任务的调用,当调用完成后,容器又启了一个连接线程,来返回最终的执行结果。


这种异步调用,在容器线程资源非常宝贵的时候,能够大大的提高整个系统的吞吐量


ps:异步调用可以使用AsyncHandlerInterceptor进行拦截,使用示例如下:

[java]  view plain  copy
  1. @Component  
  2. public class MyAsyncHandlerInterceptor implements AsyncHandlerInterceptor {  
  3.       
  4.     private static final Logger logger = LoggerFactory.getLogger(MyAsyncHandlerInterceptor.class);  
  5.   
  6.     @Override  
  7.     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)  
  8.             throws Exception {  
  9.         return true;  
  10.     }  
  11.   
  12.     @Override  
  13.     public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,  
  14.             ModelAndView modelAndView) throws Exception {  
  15. //      HandlerMethod handlerMethod = (HandlerMethod) handler;  
  16.         logger.info(Thread.currentThread().getName()+ "服务调用完成,返回结果给客户端");  
  17.     }  
  18.   
  19.     @Override  
  20.     public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)  
  21.             throws Exception {  
  22.         if(null != ex){  
  23.             System.out.println("发生异常:"+ex.getMessage());  
  24.         }  
  25.     }  
  26.   
  27.     @Override  
  28.     public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response, Object handler)  
  29.             throws Exception {  
  30.           
  31.         // 拦截之后,重新写回数据,将原来的hello world换成如下字符串  
  32.         String resp = "my name is chhliu!";  
  33.         response.setContentLength(resp.length());  
  34.         response.getOutputStream().write(resp.getBytes());  
  35.           
  36.         logger.info(Thread.currentThread().getName() + " 进入afterConcurrentHandlingStarted方法");  
  37.     }  
  38.   
  39. }  

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值