其實(shí)這個(gè)話題是緊接著上一篇文章的。
再接著講spring AOP API 的坑
出問(wèn)題的配置
//配置1
<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">
<property name="proxyTargetClass" value="true"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>
此配置的目的是想進(jìn)行cglib類代理。但是實(shí)際上當(dāng)進(jìn)行直接注入類,而不是接口時(shí)會(huì)找不到Bean錯(cuò)誤。
但是如果是這樣配置:
//配置2
<aop:aspectj-autoproxy proxy-target-class="true"/>
<tx:annotation-driven transaction-manager="transactionManager"/>
此配置可以很好的工作,并注入類(不是接口)。
分析
1、<aop:aspectj-autoproxy proxy-target-class="true"> 該命名空間會(huì)交給org.springframework.aop.config.AopNamespaceHandler處理:
registerBeanDefinitionParser("aspectj-autoproxy", new AspectJAutoProxyBeanDefinitionParser());
在AspectJAutoProxyBeanDefinitionParser中,會(huì)執(zhí)行parse方法解析配置:
public BeanDefinition parse(Element element, ParserContext parserContext) {
AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);
extendBeanDefinition(element, parserContext);
return null;
}
其中AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);目的是注冊(cè)AnnotationAwareAspectJAutoProxyCreator:
return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);
但是注意了:
if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
if (!cls.getName().equals(apcDefinition.getBeanClassName())) {
int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());
int requiredPriority = findPriorityForClass(cls);
if (currentPriority < requiredPriority) {
apcDefinition.setBeanClassName(cls.getName());
}
}
return null;
}
大家可以看到一句話:
- if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME))
- AUTO_PROXY_CREATOR_BEAN_NAME=“org.springframework.aop.config.internalAutoProxyCreator”,
- 即首先判斷當(dāng)前容器中是否包含名字為AUTO_PROXY_CREATOR_BEAN_NAME的Bean, 如果包含:然后判斷優(yōu)先級(jí),誰(shuí)優(yōu)先級(jí)高誰(shuí)獲勝,即最后那個(gè)獲勝的是實(shí)際的AutoProxyCreator
到此我們可以看到跟"<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">"配置沒(méi)什么區(qū)別,除了沒(méi)有名字外。
2、接下來(lái)看一下<tx:annotation-driven>:
該命名空間交給org.springframework.transaction.config.TxNamespaceHandler處理:
registerBeanDefinitionParser("annotation-driven", new AnnotationDrivenBeanDefinitionParser());
其中<annotation-driven> 會(huì)交給AnnotationDrivenBeanDefinitionParser進(jìn)行解析:
public BeanDefinition parse(Element element, ParserContext parserContext) {
String mode = element.getAttribute("mode");
if ("aspectj".equals(mode)) {
// mode="aspectj"
registerTransactionAspect(element, parserContext);
}else {
// mode="proxy"
AopAutoProxyConfigurer.configureAutoProxyCreator(element, parserContext);
}
return null;
}
默認(rèn)mode="proxy",所以走AopAutoProxyConfigurer.configureAutoProxyCreator,其代碼中第一句話是:
AopNamespaceUtils.registerAutoProxyCreatorIfNecessary(parserContext, element);
public static void registerAutoProxyCreatorIfNecessary(
ParserContext parserContext, Element sourceElement) {
BeanDefinition beanDefinition = AopConfigUtils.registerAutoProxyCreatorIfNecessary(
parserContext.getRegistry(), parserContext.extractSource(sourceElement));
useClassProxyingIfNecessary(parserContext.getRegistry(), sourceElement);
registerComponentIfNecessary(beanDefinition, parserContext);
}
AopConfigUtils.registerAutoProxyCreatorIfNecessary是:
registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
private static BeanDefinition registerOrEscalateApcAsRequired(Class cls, BeanDefinitionRegistry registry, Object source) {
Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {
BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);
if (!cls.getName().equals(apcDefinition.getBeanClassName())) {
int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());
int requiredPriority = findPriorityForClass(cls);
if (currentPriority < requiredPriority) {
apcDefinition.setBeanClassName(cls.getName());
}
}
return null;
}
//省略
}
此處我們又看到了registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME),如果是:
- 配置1,那么實(shí)際是兩個(gè)AutoProxyCreator。前面<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">定義了一個(gè)AutoProxyCreator,后面的<tx:annotation-driven transaction-manager="transactionManager"/>也定義另一個(gè)AutoProxyCreator。
- 配置2,那么實(shí)際是共用一個(gè)AutoProxyCreator;
而且如果配置1時(shí),因?yàn)槲覀儧](méi)有指定<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/> 所以是JDK動(dòng)態(tài)代理,因此不管怎么樣,都無(wú)法注入類的。
問(wèn)題找到了,原因是注冊(cè)了兩個(gè)AutoProxyCreator,造成了二次代理引發(fā)的問(wèn)題,這個(gè)和之前的《spring的bean二次代理問(wèn)題》一樣
如何解決
- 給配置1起名字為”org.springframework.aop.config.internalAutoProxyCreator“;
- 或者使用配置2
建議
1、如果沒(méi)有必要,請(qǐng)不要使用低級(jí)別API,如上述-->自己去創(chuàng)建AutoProxyCreator
2、首先選擇使用如:
<aop:config>或者是<org.springframework.aop.config.internalAutoProxyCreator>
如上配置已經(jīng)非常好了,根本沒(méi)必要使用低級(jí)別API。
如<tx:annotation-driven>使用的AutoProxyCreator都是和上邊是一樣的。這樣還能防止二次代理。
聲明式/@AspectJ風(fēng)格的AOP都非常好了,完全沒(méi)必要使用低級(jí)別API,請(qǐng)不要再使用低級(jí)別API了。
如何用過(guò)shiro的朋友都應(yīng)該知道如下配置:
<!-- Enable Shiro Annotations for Spring-configured beans. Only run after -->
<!-- the lifecycleBeanProcessor has run: -->
<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor"/>
<bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">
<property name="securityManager" ref="securityManager"/>
</bean>
其實(shí)我們可以換成這樣:
<aop:config proxy-target-class="true"/>
<bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">
<property name="securityManager" ref="securityManager"/>
</bean>
或者使用<aop:aspectj-autoproxy>也行,這樣也不會(huì)存在二次代理的問(wèn)題。
至于shiro這個(gè)之前的配置容易發(fā)生二次注入,我會(huì)再下一篇文章《在springmvc+shiro項(xiàng)目中使用druid監(jiān)聽(tīng)controller層出現(xiàn)IllegalStateException異常》詳細(xì)講到