当前位置: 首页 > news >正文

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)。
http://www.jsqmd.com/news/119928/

相关文章:

  • 通达信支撑线
  • RFSOC学习记录(四)MTS时序分析
  • LLM参数: Temperature 与 Top-p解析
  • LLM参数: Temperature 与 Top-p解析
  • PHP 8.5 新特性 闭包可以作为常量表达式了
  • Agent设计模式与工程化
  • 为 AI 智能体打造高效的上下文工程 -- Anthropic
  • 《深度测评:从 GPT-5.1 到 GPT-5.2,OpenAI 到底在 Pro 模型里藏了什么黑科技?》
  • nRF54LV10A 介绍
  • 做人
  • 20251221——读后感8
  • 0-1 从零开始到实现arp 断网
  • python:报错:ModuleNotFoundError: No module named langgraph
  • 全球股市估值与太空采矿技术的经济可行性
  • 自动化批量生产英语单词短视频
  • 开源 + 国产芯片:具身智能的 DeepSeek 时刻来了?
  • 9 个降AI率工具,研究生高效降AIGC指南
  • 【节点】[GammaToLinearSpaceExact节点]原理解析与实际应用
  • 20251221
  • day 36
  • AT_agc061_c [AGC061C] First Come First Serve
  • Thinkphp和Laravel全家桶鲜花售卖商城系统vue
  • 开源AI模型趋势与技术周报:声音克隆、架构转换与智能眼镜
  • gemini简易打开方式(非支持地区简短对话)
  • Thinkphp和Laravel失物招领系统vue-
  • Springboot文档管理系统 yb510(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
  • 数字员工与熊猫智汇是什么?主要特点和企业应用有哪些?
  • Vue视差标题背景
  • Thinkphp和Laravel奇思妙享博客文章新闻分享系统echart-vue
  • Thinkphp和Laravel人才求职招聘网站系统4b152