关于Spring注解@Async引发其他注解失效
概述
在前面一篇文章中,介绍,在一个Bean中注入自己,如果有@Async和@Transaction,如果使用@Autowire注入自身,会报循环依赖,如果使用BeanFactoryAware注入自己,会使得@Transaction失效。 例如:
@Service
public class MyService implements BeanFactoryAware{
private MyService self; //事务注解无效
@Transactional
public void notWork() {
...
} @Async
public Future async(){
...
} @Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
self= beanFactory.getBean(MyService.class);
}
}
当时只是简单提了一下,这篇文章就是来介绍为什么会失效。
一般情况
造成上面的情况需要满足以下条件:
- 有@Async和其他类似@Transaction注解
- 自己类在BeanFactoryAware中,通过BeanFactory获取自己
造成的结果:除@Async外的注解生效,其他的都不生效,如下图

而正常代理的应该是下图:

原因
首先想到的是@Async注解的处理方式可能和其他的不一样。在AsyncAnnotationBeanPostProcessor的实现中(具体代码是在其父类AbstractAdvisingBeanPostProcessor),发现一个问题,
正常情况下,进来的bean已经是被代理的动态代理类,而失效的时候,进来的确实实际的类,如下图:

然后在分析下代码,如果是实际的类,走到69行的时候,返回是true,把@Aysnc的Advisor加入到动态道理中,而如果是实际的类,走到83行的时候,就会创建代理类,只把@Aysnc的advisor加入到动态代理中,所已诸如@Transaction就会失效。
为什么进来的不是代理类
其实唯一的区别就是BeanFactoryAware中,是否通过了BeanFactory获取了自己。那为什么使用BeanFactory获取了自己,后续的BeanPostProcessor中就不是代理了?如果熟悉Spring @Transaction加载机制的就知道,诸如@Transaction,@Retryable 注解的动态代理创建是在AnnotationAwareAspectJAutoProxyCreator中创建的。通过debug发现,经过AnnotationAwareAspectJAutoProxyCreator后,我们的动态代理竟然没有加上。
再看一下AnnotationAwareAspectJAutoProxyCreator中的实现,但是经过他却没有生成代理类。原因竟然是提前暴露的Map里面竟然有“myService”,

他是什么时候暴露出来的呢?其实就是在
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
self= beanFactory.getBean(MyService.class);
}
那么,一切水落石出了,在实例MyService中,触发了BeanFactoryAware,通过beanFactory.getBean(MyService.class);中创建了代理类(tips:当前代理类并没有包含@Async的Adivisor),因为现在Spring其实正是在创建MyService这个Bean,还没有放入到BeanFactory中。然后我们再这个过程中又触发了一次beanFactory.getBean(MyService.class);导致创建代理并返回后,加入到了到了提前暴露的map中。导致后面的一系列问题。感觉有点绕。看图说话:
正常情况,应该是如下流程:

异常情况却是这样的

小结
正常情况下,还是使用@Autowire来注入把(如果使用Autowire,上述情况直接回抛出循环依赖)。当然,出现了问题,也是不能放过了,要知其然还要知其所以然!
关于Spring注解@Async引发其他注解失效的更多相关文章
- 使用Spring的@Async创建异步方法
使用Spring的@Async创建异步方法 在开发系统的过程中,通常会考虑到系统的性能问题,提升系统性能的一个重要思想就是“串行”改“并行”.说起“并行”自然离不开“异步”,今天我们就来聊聊如何使用S ...
- Spring @Scheduled @Async联合实现调度任务(2017.11.28更新)
定时任务之前一直用的是quartz之类,但是注意到Spring中其实也提供了一种简单的调度注释@Scheduled,也就想尝一下鲜.. 代码示意如下: @Component @EnableSchedu ...
- Spring使用@Async注解
本文讲述@Async注解,在Spring体系中的应用.本文仅说明@Async注解的应用规则,对于原理,调用逻辑,源码分析,暂不介绍.对于异步方法调用,从Spring3开始提供了@Async注解,该注解 ...
- Spring同一个类中的注解方法调用AOP失效问题总结
public interface XxxService { // a -> b void a(); void b(); } @Slf4j public class XxxServiceImpl ...
- 关于Dubbo和Spring异步注解@Async的冲突
项目中难免会有异步处理的需求,像异步记录日志啦,异步发送邮件啦,而Dubbo又是现在主流的分布式框架,所有异步+Dubbo的组合是再所难免的 但博主是实践中发现Dubbo的服务并不能很好的跟Sprin ...
- 如何在项目中使用Spring异步调用注解@Async
本文主要介绍如何使用Spring框架提供的异步调用注解@Async,异步线程池配置.异常捕获处理. 开启@Async注解支持 使用@Async注解的之前,必须在项目中启动时调用@EnableAsync ...
- Spring中异步注解@Async的使用、原理及使用时可能导致的问题
前言 其实最近都在研究事务相关的内容,之所以写这么一篇文章是因为前面写了一篇关于循环依赖的文章: <面试必杀技,讲一讲Spring中的循环依赖> 然后,很多同学碰到了下面这个问题,添加了S ...
- Spring Cloud @HystrixCommand和@CacheResult注解使用,参数配置
使用Spring Cloud时绕不开Hystrix,他帮助微服务实现断路器功能.该框架的目标在于通过控制那些访问远程系统.服务和第三方库的节点,从而对延迟和故障提供更强大的容错能力.Hystrix具备 ...
- Spring(四)使用注解注入Bean
注解简单介绍 是代码里面的特殊标记,使用注解完成功能. 注解写法@ 注解名称(属性名=属性值). 注解可以作用在类.方法.属性上面. 使用流程: 在ApplicationContext.xml中开启注 ...
随机推荐
- RTlinux3.2安装
( 1 ).前言 2003 年以后, fmslabs 的 RTLinux Free 版本为 3.2Pre ,和以前的 RTLinux 3.1 比较,不再需要必须从 2.4.4 的内核上安装. RTLi ...
- 数据结构--hashtable(散列表)
散列 散列又叫hash.是通过关键字把数据映射到指定位置的一种数据结构.理想的散列表,是一个包含关键字的固定大小的数组 哈希表存储的是键值对,其查找的时间复杂度与元素数量多少无关,哈希表在查找元素时是 ...
- javascript 一些关于css操作的函数
// 通过样式表 获得css样式 //obj 表示dom对象,name 表示css属性 比如width等 function getStyle(obj,name){ if(obj.currentStyl ...
- json_encode处理json数据中文乱码
今天使用json_encode 把一个数组转换成json数据,echo处理的时候,显示为null.去查了php手册. 该函数只能接受 UTF-8 编码的数据. 在网上搜了下,找到了解决方法. < ...
- Count:858org.apache.jasper.JasperException: Unable to compile class for JSP
1.错误描述 Count:858org.apache.jasper.JasperException: Unable to compile class for JSP: An error occurre ...
- js、css动态压缩页面代码
1.js.css动态压缩页面代码 <%@ Page Language="C#" AutoEventWireup="true" CodeFile=" ...
- 自定义方法实现ArrayList排序
package cn.edu.nwpu.java; import java.util.ArrayList; import java.util.Collection; import com.sun.xm ...
- Good Bye 2017 E. New Year and Entity Enumeration
先按照绿点进行分块 第一个绿点和最后一个绿点之后很好处理不说了 两个绿点之间的讨论: 有两种方案 1:红(蓝)点和绿点顺序连接,距离为相邻绿点距离(也就是双倍绿点距离) 2:红(蓝)点和绿点的点阵中寻 ...
- 如何登录mysql? cmd怎么连接mysql数据库||从MYSQL客户端登录MYSQL
1 2 3 4 5 6 7 分步阅读 Mysql开源数据库,任何人都可以下载安装使用.那么安装好的mysql如何登陆连接mysql数据库呢?本经验咗嚛介绍几种常见的方法 工具/原料 mysql 连 ...
- Defeat the Enemy UVALive - 7146
Long long ago there is a strong tribe living on the earth. They always have wars and eonquer other ...