Spring到底何时为使用事务注解的bean创建代理 ?

今天说这个问题之前,大家需要知道一个热知识,那就是一个类中的某个方法被赋予注解@Transactional,则无法在本类中进行调用,否则事务失效,那是因为,无法创建代理类,也就是无法使用事务特性,毕竟spring的事务是通过AOP中的代理模式进行实现的,那么问题来了?spring的bean何时被创建代理类呢,又怎么知道的呢,如果@Transactional放在类上,那就可以理解bean在进行实例化的时候就知道了,但是放在方法上呢?其实也是通过反射的方式给这个对象进行生成代理类的,接下来就重点讲解下这个问题

  1. 应用启动过程中Spring AOP自动配置机制AopAutoConfiguration执行

仅在注解 @EnableAspectJAutoProxy 类存在于classpath并且 spring.aop 没有明确设置为 false 时应用
可以认为等价于主动使用注解 @EnableAspectJAutoProxy

1.1 注解@EnableAspectJAutoProxy导入了类AspectJAutoProxyRegistrar

AspectJAutoProxyRegistrar是一个ImportBeanDefinitionRegistrar,

1.2 AspectJAutoProxyRegistrar向容器注册一个AnnotationAwareAspectJAutoProxyCreator

长类名 : org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator
这是一个BeanPostProcessor,它会在每个bean实例创建时检查该bean是否需要创建代理,需要的话
就创建相应的代理。

  1. 应用启动过程中自动配置机制TransactionAutoConfiguration执行

TransactionAutoConfiguration使用了注解@EnableTransactionManagement

2.1 注解@EnableTransactionManagement导入TransactionManagementConfigurationSelector
2.2 TransactionManagementConfigurationSelector根据参数和包引用状态决定引入 :
2.2.1 ProxyTransactionManagementConfiguration (PROXY)或者AspectJJtaTransactionManagementConfiguration(ASPECTJ,使用JTA) 或者
AspectJTransactionManagementConfiguration(ASPECTJ,不使用JTA)

ProxyTransactionManagementConfiguration是较常见的一种情况,这里我们以此为例继续分析。

2.3 ProxyTransactionManagementConfiguration配置类定义bean : AnnotationTransactionAttributeSource

AnnotationTransactionAttributeSource用于分析理解事务注解属性@Transactional的语义

2.4 ProxyTransactionManagementConfiguration配置类定义bean : TransactionInterceptor

属性 transactionAttributeSource : bean AnnotationTransactionAttributeSource

2.5 ProxyTransactionManagementConfiguration配置类定义bean : BeanFactoryTransactionAttributeSourceAdvisor

类型 : BeanFactoryTransactionAttributeSourceAdvisor, 名称:transactionAdvisor
属性 advice : bean TransactionInterceptor
属性 transactionAttributeSource : bean AnnotationTransactionAttributeSource

  1. 某个带有事务注解属性的服务组件bean(带注解@Component)首次被使用时实例化

因为AnnotationAwareAspectJAutoProxyCreator是一个BeanPostProcessor,所以它在每个bean实例化时会针对该bean执行,所以在带有事务注解属性的服务组件bean被实例化时,它也被应用;
服务组件bean如果在类或者方法级别使用了事务注解@Transactional,则它会被AnnotationAwareAspectJAutoProxyCreator认为是需要创建代理(用于进行事务处理),否则不需要创建代理 ;
AnnotationAwareAspectJAutoProxyCreator认为某个服务组件bean需要创建代理对象时,为其创建代理。
具体代码参考AnnotationAwareAspectJAutoProxyCreator基类AbstractAutoProxyCreator:

// AnnotationAwareAspectJAutoProxyCreator 基类 AbstractAutoProxyCreator 代码片段
// 接口 BeanPostProcessor 定义的方法
@Override
public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
	if (bean != null) {
		Object cacheKey = getCacheKey(bean.getClass(), beanName);
		if (!this.earlyProxyReferences.contains(cacheKey)) {
			return wrapIfNecessary(bean, beanName, cacheKey);
		}
	}
	return bean;
}

protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {
	if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {
		return bean;
	}
	if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {
		return bean;
	}
	if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {
		this.advisedBeans.put(cacheKey, Boolean.FALSE);
		return bean;
	}

	// Create proxy if we have advice.
	// 使用了事务注解的bean上的事务注解@Transactional会被发现,
	// 进而发现 bean BeanFactoryTransactionAttributeSourceAdvisor 到 specificInterceptors 
	// 具体的发现逻辑可以参考工具类方法 : AopUtils#findAdvisorsThatCanApply
	Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
	// 如果 specificInterceptors 不为空,说明需要有 advise 需要被应用到该bean,所以需要创建相应的代理
	// 对象,外面包裹相应的 advise : specificInterceptors 。
	// 这里 DO_NOT_PROXY 是一个常量,值为 null
	if (specificInterceptors != DO_NOT_PROXY) {
		this.advisedBeans.put(cacheKey, Boolean.TRUE);
		// 创建代理
		// 原bean对象 : bean
		// 代理对象 : proxy
		Object proxy = createProxy(
				bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));
		this.proxyTypes.put(cacheKey, proxy.getClass());
		return proxy;
	}

	this.advisedBeans.put(cacheKey, Boolean.FALSE);
	return bean;
}

相关文章
Spring Transaction 5.1.2 源代码分析 : AnnotationTransactionAttributeSource 解析注解式事务属性
Spring BeanPostProcessor : AnnotationAwareAspectJAutoProxyCreator

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值