Spring Boot 使用WebAsyncTask异步返回结果
在Spring Boot中(Spring MVC)下请求默认都是同步的,一个请求过去到结束都是由一个线程负责的,很多时候为了能够提高吞吐量,需要将一些操作异步化,除了一些耗时的业务逻辑可以异步化,我们的查询接口也是可以做到异步执行。
一个请求到服务上,是用的web容器的线程接收的,比如线程http-nio-8084-exec-1
我们可以使用WebAsyncTask将这个请求分发给一个新的线程去执行,http-nio-8084-exec-1可以去接收其他请求的处理。一旦WebAsyncTask返回数据有了,就会被再次调用并且处理,以异步产生的方式,向请求端返回值。
示例代码如下:
@RequestMapping(value="/login", method = RequestMethod.GET) public WebAsyncTask<ModelAndView> longTimeTask(){ System.out.println("/login被调用 thread id is : " + Thread.currentThread().getName()); Callable<ModelAndView> callable = new Callable<ModelAndView>() { public ModelAndView call() throws Exception { Thread.sleep(1000); /模拟长时间任务 ModelAndView mav = new ModelAndView("login/index"); System.out.println("执行成功 thread id is : " + Thread.currentThread().getName()); return mav; } }; return new WebAsyncTask<ModelAndView>(callable); }
可以看到输出结果如下:
/login被调用 thread id is : http-nio-8084-exec-1
执行成功 thread id is : MvcAsync1
在执行业务逻辑之前的线程和具体处理业务逻辑的线程不是同一个,达到了我们的目的。
然后我做了一个并发测试,发现不停的在创建MvcAsync1这个线程,我就在想,难道没有用线程池?
通过阅读源码才发现果真如此,WebAsyncManager是Spring MVC管理async processing的中心类。
默认是使用SimpleAsyncTaskExecutor,这个会为每次请求创建一个新的线程
private AsyncTaskExecutor taskExecutor = new SimpleAsyncTaskExecutor(this.getClass().getSimpleName());
如果说任务指定了executor,就用任务指定的,没有就用默认的SimpleAsyncTaskExecutor
AsyncTaskExecutor executor = webAsyncTask.getExecutor(); if (executor != null) { this.taskExecutor = executor; }
我们可以配置async 的线程池,不需要为每个任务单独指定
通过configurer.setTaskExecutor(threadPoolTaskExecutor());来指定
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; import org.springframework.web.context.request.async.TimeoutCallableProcessingInterceptor; import org.springframework.web.servlet.config.annotation.AsyncSupportConfigurer; import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; @Configuration public class WebMvcConfig extends WebMvcConfigurationSupport { @Override public void configureAsyncSupport(final AsyncSupportConfigurer configurer) { configurer.setDefaultTimeout(60 * 1000L); configurer.registerCallableInterceptors(timeoutInterceptor()); configurer.setTaskExecutor(threadPoolTaskExecutor()); } @Bean public TimeoutCallableProcessingInterceptor timeoutInterceptor() { return new TimeoutCallableProcessingInterceptor(); } @Bean public ThreadPoolTaskExecutor threadPoolTaskExecutor() { ThreadPoolTaskExecutor t = new ThreadPoolTaskExecutor(); t.setCorePoolSize(10); t.setMaxPoolSize(50); t.setThreadNamePrefix("YJH"); return t; } }
配置完之后就可以看到输出的线程名称是YJH开头的了,而且也不会一直创建新的线程
可以看到输出结果如下:
/login被调用 thread id is : http-nio-8084-exec-1 执行成功 thread id is : YJH1
总结
以上所述是小编给大家介绍的Spring Boot 使用WebAsyncTask异步返回结果,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!
您可能感兴趣的文章:
- 深入理解spring boot异步调用方式@Async
- spring boot 使用@Async实现异步调用方法
- spring boot中使用@Async实现异步调用任务
- spring boot使用自定义配置的线程池执行Async异步任务
- spring boot使用自定义的线程池执行Async任务
- spring boot异步(Async)任务调度实现方法
- Spring Boot集成教程之异步调用Async
- Spring Boot利用@Async如何实现异步调用:自定义线程池
- Spring Boot利用@Async异步调用:ThreadPoolTaskScheduler线程池的优雅关闭详解
- Spring Boot利用@Async异步调用:使用Future及定义超时详解
- java多线程Future、FutureTask使用示例,返回异步的结果
- Python学习之使用Future对象来异步返回结果方法详解
- java多线程Future、FutureTask使用示例,返回异步的结果
- Android 中 Handler 引起的内存泄露 在Android常用编程中,Handler在进行异步操作并处理返回结果时经常被使用。其实这可能导致内存泄露,代码中哪里可能导致内存泄露,又是如何
- C#使用BeginInvoke和EndInvoke异步下载和获取返回结果
- python里使用Future对象来异步返回结果
- 获取Executor提交的并发执行的任务返回结果的两种方式/ExecutorCompletionService使用
- DEDE使用AJAX无刷新提交Form表单,PHP返回结果
- 使用QFuture类监控异步计算的结果
- Android使用ZXing扫描二维码,并返回二维码结果
- javascript中使用ajax异步验证数据库后方法return true时总是返回undefined
- 使用Retrofit打印请求日志,过滤改变服务器返回结果,直接获取String字符串
- list中remove的使用方法及返回结果的不同
- 使用MyBatis指定返回结果与实体属性相对应的方法
- 【Java并发】- 使用CompletionService异步收集任务结果
- Android使用意图实现返回结果
- JAVA 使用HttpClient 实现简单发送HTTP请求 无返回结果
- 多线程 : 使用 CompletionService 多线程返回结果
- 获取Executor提交的并发执行的任务返回结果的两种方式/ExecutorCompletionService使用
- 使用扩展方法对Linq to SQl Classes扩展,使其在增删改时返回结果