0.写在前面
上一篇文章排查了@Async失效问题,于是就想弄清楚@Async实现原理,发现这篇文章对源码讲解比较清楚,于是参考它的源码分析思路,总结成自己可以理解的内容。
1.背景
大家项目中用到异步、多线程的场景很多,使用最多的场景还是主动对象模式,就是主线程开启一个线程池去任务分发,任务执行完成之后,关闭线程池,但是有的场景则需要部分代码异步执行的效果,简单的说就是有一个可以复用的线程池可以复用,直接new Thread当然也可以,不推荐,没有线程池的可靠性好,如果这个时候再创建一个线程池用完再关闭代码是不是有点重,维护性也不好,@Async这个注解就是为了解决这个问题,只需要在bean的方法上加上这和注解即可,就可以实现异步。
2正文
先简单介绍下怎么使用
1、先在程序上加上@EnableAsync这个注解,可以指定执行器,如果不指定就从beanFactory中获取taskExecutor这个执行器
@Async(value = "executor")
public void test(){
2、如果想自定义执行器,实现这个接口org.springframework.scheduling.annotation.AsyncConfigurer
或者继承org.springframework.scheduling.annotation.AsyncConfigurerSupport
这个类
public class AsyncExecutor implements AsyncConfigurer {
@Override
public Executor getAsyncExecutor() {
// 自定义线程池
return Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());
}
源码实现
1.@EnableAsync这个注解做什么了什么事情
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(AsyncConfigurationSelector.class)
public @interface EnableAsync {
可以看到@Import引入了AsyncConfigurationSelector这个类,前面介绍spring ioc源码解析的时候这个@Import这个注解会把引入的类进行bean定义解析后注册到beanFactory而后进行实例化。
我们看到这个selector又继承了这个类
public class AsyncConfigurationSelector extends AdviceModeImportSelector<EnableAsync> {
spring切面拦截到这个注解后进行下一步操作。
org.springframework.scheduling.annotation.AsyncConfigurationSelector#selectImports
@Override
public String[] selectImports(AdviceMode adviceMode) {
switch (adviceMode) {
case PROXY:
return new String[] {
ProxyAsyncConfiguration.class.getName() };
case ASPECTJ:
return new String[] {
ASYNC_EXECUTION_ASPECT_CONFIGURATION_CLASS_NAME };
default:
return null;
}