SpringBoot使用@Async注解实现异步处理

在使用SpringBoot开发中,有些业务时间处理比较久,需要新开线程实现业务的异步处理。通常的做法是建立一个线程池,然后把需要异步处理的任务交给线程池处理。在SpringBoot上,可以不需要这么麻烦,只需要在需要异步处理的方法上加一个简单的注解@Async即可实现异步操作。

这里有一个要注意的是,不要忘记在启动类上增加@EnableAsync注解,缺少这个注解异步将不会生效。

@SpringBootApplication
@EnableAsync
public class DemoApplication {

    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }

}

这里举一个小例子进行演示,方法内仅打印一下当前线程名:

@Service
public class TestService {

    public void test() {
        System.out.println("同步方法使用线程");
        System.out.println(Thread.currentThread().getName());
        System.out.println("-------------");
    }

    @Async
    public void asyncTest() {
        System.out.println("异步方法使用线程");
        System.out.println(Thread.currentThread().getName());
        System.out.println("-------------");
    }

}

触发异步的web请求:

@RestController
@RequestMapping("/")
public class TestController {

    @Autowired
    private TestService testService;

    @GetMapping("/test")
    public String test() {
        System.out.println("web请求使用线程");
        System.out.println(Thread.currentThread().getName());
        System.out.println("-------------");
        testService.test();
        testService.asyncTest();
        return "test";
    }

}

结果:

可以看到异步注解@Async生效了。使用的是不同的线程。

@Async本质上还是使用的线程池,只是SpringBoot帮我们简化了很多操作,那如果我们需要自定义异步任务使用的线程池呢?也是可以实现,只是我们需要一个配置类。

/**
 * 线程池配置
 * AsyncConfigurer的两个方法是@Async默认线程池的配置
 */
@Configuration
public class ThreadPoolConfig implements AsyncConfigurer {

    @Override
    public Executor getAsyncExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        // 设置核心线程数
        executor.setCorePoolSize(2);
        // 设置最大线程数
        executor.setMaxPoolSize(128);
        // 设置默认线程名称
        executor.setThreadNamePrefix("thread-AsyncExecutor-");
        // 设置拒绝策略rejection-policy:当pool已经达到max size的时候,如何处理新任务 CALLER_RUNS:不在新线程中执行任务,而是由调用者所在的线程来执行
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        // 等待所有任务结束后再关闭线程池
        executor.setWaitForTasksToCompleteOnShutdown(true);
        // 设置非核心线程超时回收时间
        executor.setKeepAliveSeconds(60);
        executor.initialize();
        return executor;
    }

    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return null;
    }

    /**
     * 自定义线程池样例
     * @return
     */
    @Bean("example")
    public Executor example() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        // 设置核心线程数
        executor.setCorePoolSize(2);
        // 设置最大线程数
        executor.setMaxPoolSize(128);
        // 设置默认线程名称
        executor.setThreadNamePrefix("thread-example-");
        // 设置拒绝策略rejection-policy:当pool已经达到max size的时候,如何处理新任务 CALLER_RUNS:不在新线程中执行任务,而是由调用者所在的线程来执行
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        // 等待所有任务结束后再关闭线程池
        executor.setWaitForTasksToCompleteOnShutdown(true);
        // 设置非核心线程超时回收时间
        executor.setKeepAliveSeconds(60);
        return executor;
    }

}

配置类实现了AsyncConfigurer接口中的2个方法,其中getAsyncExecutor()方法用来获取@Async默认线程池,即我们如果需要自定义@Async默认线程池则直接在此方法中实例化一个线程池并返回即可;getAsyncUncaughtExceptionHandler()是用来返回对异步任务异常之后的一些处理。这里暂时没用直接返回null;如果我们项目中需要多个线程池,则可以参考example()方法实例出自定义的线程池example,在使用的时候使用@Async("example")来实现使用example的线程池实现异步操作。

使用自定义example线程池的方法示例:

@Async("example")
public void asyncExampleTest() {
    System.out.println("异步方法使用自定义example线程");
    System.out.println(Thread.currentThread().getName());
    System.out.println("-------------");
}

执行结果:

完整代码参考:https://gitee.com/floow/blog-demo demo13

支付宝搜索:344355 领取随机红包

如果文章对您有帮助,欢迎给作者打赏