Spring Boot学习(三十二):异步执行注解@Async

前言

当我们需要异步执行某个方法时,最常用的方法就是新建一个线程去执行任务或者把任务提交到线程池异步执行。在spring boot,最简单的方式就是使用 @Async注解。下面我们就来体验下 @Async注解

1、开启异步异步调用

在启动类添加@EnableAsync注解即可开启异步调用

@SpringBootApplication
@MapperScan("com.example.testdemo.mapper")
@EnableAsync
public class TestdemoApplication {

}

2、定义异步任务

使用@Async注解定义异步任务:我们只需要在一个方法上使用@Async注解,该方法就定义为异步方法了

异步的方法有3种

  1. 最简单的异步调用,返回值为void
  2. 带参数的异步调用 异步方法可以传入参数
  3. 异常调用返回Future

下面是最常见的2种异步方法

@Component
public class AsyncDemo {
    

    /**
     * 普通异步调用
     *
     * @param parameter
     */
    @Async
    public void asyncMethod(String parameter) {
        System.out.println("入参为:"+parameter);
    }

    /**
     * 异常调用返回Future
     * @param parameter
     * @return
     */
    @Async
    public Future<String> asyncFutureMethod(String parameter) {
        Future<String> future= new AsyncResult<String>("success:" + parameter);
        return future;
    }

}

常见TaskExecutor

Spring发行版中包含了许多预先构建的TaskExecutor实现,以下是spring的TaskExecutor实现类

ThreadPoolTaskExecutor

ThreadPoolTaskExecutor是使用ThreadPoolExecutor实现的,即是ThreadPoolExecutor的Spring包装。
ThreadPoolTaskExecutor同样是提供线程池执行任务,但是可以使用xml或者JavaBean的形式进行配置,初始化。

SimpleAsyncTaskExecutor

此实现不重用任何线程,而是为每次调用启动一个新线程。但是,它支持一个并发限制,该限制将阻塞任何超过该限制的调用,直到释放一个槽为止。
不是真的线程池,只是不重用线程,每次调用都会创建一个新的线程。

SyncTaskExecutor

此实现不异步执行调用。相反,每次调用都在调用线程中进行。它主要用于不需要多线程的情况下,比如在简单的测试用例中。

ConcurrentTaskExecutor

 Executor的适配类,不推荐使用。如果ThreadPoolTaskExecutor不满足要求时,才用考虑使用这个类。

Async默认使用的是 SimpleAsyncTaskExecutor

3、自定义Async配置

我们可以通过实现AsyncConfigurer来自定义一些异步配置,比如使用的线程池、异常处理

面是一个自定义的AsyncConfigurer配置,大家可以参考

@Slf4j
@Configuration
public class MyAsyncConfigurer implements AsyncConfigurer {

    /**
     * 使用的线程池
     * @return
     */
    @Override
    public Executor getAsyncExecutor() {
        ThreadPoolTaskExecutor threadPoolTaskExecutor = new ThreadPoolTaskExecutor();
        //设置核心线程数
        threadPoolTaskExecutor.setCorePoolSize(2);
        //设置最大线程数
        threadPoolTaskExecutor.setMaxPoolSize(6);
        //设置队列长度
        threadPoolTaskExecutor.setQueueCapacity(10);
        //设置超出核心线程数的空闲线程存活时间,秒为单位,默认60
        threadPoolTaskExecutor.setKeepAliveSeconds(60);
        //设置线程名前缀
        threadPoolTaskExecutor.setThreadNamePrefix("MyAsync-");
        //线程池的饱和策略 我这里设置的是 CallerRunsPolicy 也就是由用调用者所在的线程来执行任务 共有四种
        //AbortPolicy:直接抛出异常,这是默认策略;
        //CallerRunsPolicy:用调用者所在的线程来执行任务;
        //DiscardOldestPolicy:丢弃阻塞队列中靠最前的任务,并执行当前任务;
        //DiscardPolicy:直接丢弃任务;
        threadPoolTaskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        //设置在关闭线程池时是否等待任务完成
        threadPoolTaskExecutor.setWaitForTasksToCompleteOnShutdown(true);
        //设置等待终止的秒数
        threadPoolTaskExecutor.setAwaitTerminationSeconds(60);
        return threadPoolTaskExecutor;
    }

    /**
     * 异常处理
     * @return
     */
    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return (Throwable throwable, Method method, Object... obj)->{
            log.info("Method name -{}, Exception message -{}" ,method.getName(),throwable);
        };
    }

}
  • 0
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值