Spring Boot 使用WebAsyncTask异步返回结果
在SpringBoot中(SpringMVC)下请求默认都是同步的,一个请求过去到结束都是由一个线程负责的,很多时候为了能够提高吞吐量,需要将一些操作异步化,除了一些耗时的业务逻辑可以异步化,我们的查询接口也是可以做到异步执行。
一个请求到服务上,是用的web容器的线程接收的,比如线程http-nio-8084-exec-1
我们可以使用WebAsyncTask将这个请求分发给一个新的线程去执行,http-nio-8084-exec-1可以去接收其他请求的处理。一旦WebAsyncTask返回数据有了,就会被再次调用并且处理,以异步产生的方式,向请求端返回值。
示例代码如下:
@RequestMapping(value="/login",method=RequestMethod.GET) publicWebAsyncTasklongTimeTask(){ System.out.println("/login被调用threadidis:"+Thread.currentThread().getName()); Callable callable=newCallable (){ publicModelAndViewcall()throwsException{ Thread.sleep(1000);/模拟长时间任务 ModelAndViewmav=newModelAndView("login/index"); System.out.println("执行成功threadidis:"+Thread.currentThread().getName()); returnmav; } }; returnnewWebAsyncTask (callable); }
可以看到输出结果如下:
/login被调用threadidis:http-nio-8084-exec-1
执行成功threadidis:MvcAsync1
在执行业务逻辑之前的线程和具体处理业务逻辑的线程不是同一个,达到了我们的目的。
然后我做了一个并发测试,发现不停的在创建MvcAsync1这个线程,我就在想,难道没有用线程池?
通过阅读源码才发现果真如此,WebAsyncManager是SpringMVC管理asyncprocessing的中心类。
默认是使用SimpleAsyncTaskExecutor,这个会为每次请求创建一个新的线程
privateAsyncTaskExecutortaskExecutor=newSimpleAsyncTaskExecutor(this.getClass().getSimpleName());
如果说任务指定了executor,就用任务指定的,没有就用默认的SimpleAsyncTaskExecutor
AsyncTaskExecutorexecutor=webAsyncTask.getExecutor(); if(executor!=null){ this.taskExecutor=executor; }
我们可以配置async的线程池,不需要为每个任务单独指定
通过configurer.setTaskExecutor(threadPoolTaskExecutor());来指定
importorg.springframework.context.annotation.Bean; importorg.springframework.context.annotation.Configuration; importorg.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; importorg.springframework.web.context.request.async.TimeoutCallableProcessingInterceptor; importorg.springframework.web.servlet.config.annotation.AsyncSupportConfigurer; importorg.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; @Configuration publicclassWebMvcConfigextendsWebMvcConfigurationSupport{ @Override publicvoidconfigureAsyncSupport(finalAsyncSupportConfigurerconfigurer){ configurer.setDefaultTimeout(60*1000L); configurer.registerCallableInterceptors(timeoutInterceptor()); configurer.setTaskExecutor(threadPoolTaskExecutor()); } @Bean publicTimeoutCallableProcessingInterceptortimeoutInterceptor(){ returnnewTimeoutCallableProcessingInterceptor(); } @Bean publicThreadPoolTaskExecutorthreadPoolTaskExecutor(){ ThreadPoolTaskExecutort=newThreadPoolTaskExecutor(); t.setCorePoolSize(10); t.setMaxPoolSize(50); t.setThreadNamePrefix("YJH"); returnt; } }
配置完之后就可以看到输出的线程名称是YJH开头的了,而且也不会一直创建新的线程
可以看到输出结果如下:
/login被调用threadidis:http-nio-8084-exec-1 执行成功threadidis:YJH1
总结
以上所述是小编给大家介绍的SpringBoot使用WebAsyncTask异步返回结果,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!