美文网首页java编程世界互联网科技
Spring Boot 使用WebAsyncTask异步返回结果

Spring Boot 使用WebAsyncTask异步返回结果

作者: 尹吉欢 | 来源:发表于2018-02-02 13:50 被阅读389次

    在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
    

    相关文章

      网友评论

      • jimmyinfo:请问:为啥不增加tomcat的http连接线程池数量,反而要内部新增加一个线程池,使用这些线程来增加系统吞吐能力?

      本文标题:Spring Boot 使用WebAsyncTask异步返回结果

      本文链接:https://www.haomeiwen.com/subject/btxezxtx.html