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/lqccan/blog-demo demo13


原网址: 访问
创建于: 2021-06-25 13:52:45
目录: default
标签: 无

请先后发表评论
  • 最新评论
  • 总共0条评论