这篇接事务的分析总篇,上文介绍了TransactionManagementConfigurationSelector导入两个类,其中之一就是AutoProxyRegistrar,进入到该类中,发现实现了ImportBeanDefinitionRegistrar,所以导入的还不是AutoProxyRegistrar,是要调用registerBeanDefinitions方法,手动注册bd了。
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
boolean candidateFound = false;
Set<String> annTypes = importingClassMetadata.getAnnotationTypes();
for (String annType : annTypes) {
AnnotationAttributes candidate = AnnotationConfigUtils.attributesFor(importingClassMetadata, annType);
if (candidate == null) {
continue;
}
Object mode = candidate.get("mode");
Object proxyTargetClass = candidate.get("proxyTargetClass");
if (mode != null && proxyTargetClass != null && AdviceMode.class == mode.getClass() &&
Boolean.class == proxyTargetClass.getClass()) {
candidateFound = true;
if (mode == AdviceMode.PROXY) {
AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);
if ((Boolean) proxyTargetClass) {
AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry);
return;
}
}
}
}
}
从元数据中解析出EnableTransactionManagement 注解的属性值,如果是代理的话,就执行AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);
主要的逻辑就是两块,AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);是其一,如果是要代理类的话,还要执行AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry);
下面看下AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);这里的逻辑
AopConfigUtils#registerAutoProxyCreatorIfNecessary(org.springframework.beans.factory.support.BeanDefinitionRegistry, java.lang.Object)
public static BeanDefinition registerAutoProxyCreatorIfNecessary(
BeanDefinitionRegistry registry, @Nullable Object source) {
return registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
}
这块熟悉吗,从文章中体会不到,看源码的时候应该可以体会到,这不是咱们分析过的吗,注侧AnnotationAwareAspectJAutoProxyCreator的地方吗,现在注册的变成了InfrastructureAdvisorAutoProxyCreator类。
进入到InfrastructureAdvisorAutoProxyCreator类,继承AbstractAdvisorAutoProxyCreator,好吧,豁然开朗,这玩意和AnnotationAwareAspectJAutoProxyCreator是一样的,就是找切面的。
需要注意的是它重写了方法,要找用注解@role注解标注,还是BeanDefinition.ROLE_INFRASTRUCTURE的。
@Override
protected boolean isEligibleAdvisorBean(String beanName) {
return (this.beanFactory != null && this.beanFactory.containsBeanDefinition(beanName) &&
this.beanFactory.getBeanDefinition(beanName).getRole() == BeanDefinition.ROLE_INFRASTRUCTURE);
}
回忆一下之前的找切面,只要是Advisor类型的都算,没有特殊的要求。这个类就特殊一点了。
总结一下
本类的功能就是创建代理的功能类,找切面要找@Role标注,值是BeanDefinition.ROLE_INFRASTRUCTURE的。其他逻辑和AOP逻辑一致1,找切面;2,切面是否能应用该实例上;3,返回合适的切面;4,有合适的就创建代理;