news 2026/4/12 2:43:58

Async 注解原理分析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Async 注解原理分析

核心作用:Spring 提供的注解,标注在类或方法上,使方法在异步线程中执行,调用者无需等待方法完成即可继续执行后续逻辑。

使用两步走

  1. 启动类添加@EnableAsync,开启异步任务支持;
  2. 需异步执行的方法 / 类上添加@Async注解(示例中常用CompletableFuture处理异步结果)。

@Async 原理分析

@Async可以异步执行任务,本质上是使用动态代理来实现的。通过 Spring 中的后置处理器BeanPostProcessor为使用@Async注解的类创建动态代理,之后@Async注解方法的调用会被动态代理拦截,在拦截器中将方法的执行封装为异步任务提交给线程池处理。

开启异步:@EnableAsync 的作用

  • @EnableAsync通过@Import(AsyncConfigurationSelector.class)加载配置类;
  • AsyncConfigurationSelector根据AdviceMode(默认PROXY)选择加载ProxyAsyncConfiguration
  • ProxyAsyncConfiguration注册关键后置处理器AsyncAnnotationBeanPostProcessor

后置处理器:动态代理的核心

  • AsyncAnnotationBeanPostProcessor@Async生效的关键,会为标注@Async的类创建动态代理;
  • 该处理器通过setBeanFactory()方法创建AsyncAnnotationAdvisor(Spring AOP 的Advisor组件)。

创建Advisor:定义拦截规则与逻辑

dvisor包含Advice(通知逻辑)和Pointcut(切入点):

  • Advice:创建AnnotationAsyncExecutionInterceptor拦截器,负责异步执行逻辑;
  • Pointcut:通过AnnotationMatchingPointcut匹配类 / 方法上的@Async注解(类级别 + 方法级别)。

后置处理器逻辑:生成代理对象

  • Bean 初始化后,postProcessAfterInitialization()方法判断 Bean 是否符合Advisor规则;
  • 符合规则则通过ProxyFactory创建代理对象,后续@Async方法调用会被代理拦截。

@Async 注解方法的拦截

1. 确定异步执行器(线程池)

  • 优先从缓存获取,或通过@Asyncvalue限定符查找自定义线程池;
  • 无自定义线程池时,默认使用SimpleAsyncTaskExecutor风险提示:每次创建新线程,无复用,高并发下易导致资源耗尽)。

2. 封装异步任务

  • 将目标方法执行逻辑(invocation.proceed())封装为Callable任务;
  • 若方法返回Future类型,会阻塞等待结果(处理异步嵌套调用),其他类型执行后返回null

3. 提交异步任务

根据方法返回值类型选择提交方式:

  • CompletableFuture:用supplyAsync()提交;
  • ListenableFuture:用submitListenable()提交;
  • Future:直接submit()提交;
  • void或其他类型:submit()提交后返回null

总结

理解@Async原理的核心在于理解@EnableAsync注解,该注解开启了异步任务的功能。

主要流程如下图:

@Async 使用建议

必须自定义线程池

  • 避免使用默认的SimpleAsyncTaskExecutor,推荐ThreadPoolTaskExecutor
  • 可配置核心线程数、最大线程数、队列容量等(示例:创建executor1executor2等不同线程池,通过@Async("executor1")指定)。

避免 @Async 注解实效

失效场景

原因

解决方案

同一类内调用异步方法

绕过 Spring 代理,未触发拦截

将异步方法移至另一个 Spring Bean

异步方法用static修饰

代理无法拦截静态方法(不属于实例)

用非静态包装方法调用静态逻辑

未加@EnableAsync

未开启异步支持

启动类添加@EnableAsync

方法所在类非 Spring Bean

Spring 无法创建代理

确保类被@Service等注解管理

规范返回值类型

  • 无需结果:返回void
  • 需要结果:返回Future子类(如CompletableFutureListenableFuture);
  • 其他类型(如StringObject):无法获取方法执行结果。

处理异步方法异常

  • 全局处理:实现AsyncConfigurer重写getAsyncUncaughtExceptionHandler(),自定义异常处理器;
  • 局部处理:用CompletableFutureexceptionally()等方法捕获异常。

事务管理注意

  • 异步方法需事务时,需添加@Transactional(propagation = Propagation.REQUIRES_NEW),开启独立新事务(避免与调用方事务关联)。

控制执行顺序

  • 异步方法默认无序,需按顺序执行时,用CompletableFuturethenCompose()thenAccept()等方法串联任务(如先执行fetchDataAsync,再执行processDataAsync)。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/31 12:33:43

记一次 .NET 某药品缺陷高速检测系统 卡慢分析

一:背景1. 讲故事上个月有位朋友找到我,说他们公司的程序当内存达到一定阈值(2g)之后,业务逻辑明显变慢导致下位机超时报警,想让我看下到底怎么回事,这种问题其实抓dump比较难搞,但朋友也说了有一个增长阈值…

作者头像 李华
网站建设 2026/4/10 9:10:05

全球股市估值与太空采矿技术的经济可行性

全球股市估值与太空采矿技术的经济可行性关键词:全球股市估值、太空采矿技术、经济可行性、市场分析、投资潜力摘要:本文深入探讨了全球股市估值与太空采矿技术经济可行性之间的关联。首先介绍了研究的背景、目的、预期读者和文档结构,对相关…

作者头像 李华
网站建设 2026/4/10 23:37:03

PMSM负载估计:文献复现之旅

PMSM负载估计 负载转矩预测文献复现 永磁同步电机负载转矩估计、PMSM负载转矩测量、负载预测、转矩预测的MATLAB/simulink仿真模型,模型包可运行,配套9页的英文文献,部分章节已截图。 负载估计方法包括卡尔曼滤波、离散卡尔曼滤波、Luenberge…

作者头像 李华
网站建设 2026/4/10 12:29:55

打造三菱PLC自动售货机系统:从搭建到实现

三菱PLC自动售货机系统,系统才用三菱GXworks2软件,GTdesigner3触摸屏软件编写,包含plc触摸屏,io表,原理图,流程图,接线图,报告等在自动化控制领域,三菱PLC以其稳定性和强…

作者头像 李华
网站建设 2026/3/28 16:25:29

通达信共振指标成功率80%

{}月K:"KDJ.K"(89,3,3),COLOR00FF00; 月D:"KDJ.D"(89,3,3),COLOR00FF00,LINETHICK2; 周K:"KDJ.K"(27,3,3),COLORRED; 周D:"KDJ.D"(27,3,3),COLORRED,LINETHICK2; 日K:"KD.K"(5,3,3),COLORWHITE; 日D:"KD.D"(5,3,3…

作者头像 李华
网站建设 2026/4/11 22:42:31

Thinkphp和Laravel教室租赁自习室预约vue

目录具体实现截图项目开发技术介绍PHP核心代码部分展示系统结论源码获取/同行可拿货,招校园代理具体实现截图 本系统(程序源码数据库调试部署讲解)带文档1万字以上 同行可拿货,招校园代理 Thinkphp和Laravel教室租赁自习室预约vue 项目开发技术介…

作者头像 李华