請不要再使用低級別的AOP API

在此我再給大家舉一個使用低級別AOP API遇到的坑。

 

出問題的配置

Java代碼  收藏代碼
  1. <bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">  
  2.     <property name="proxyTargetClass" value="true"/>  
  3. </bean>  
  4. <tx:annotation-driven transaction-manager="transactionManager"/>   

此配置的目的是想進行cglib類代理。但是實際上當進行直接注入類,而不是接口時會找不到Bean錯誤。

 

但是如果是這樣配置: 

Java代碼  收藏代碼
  1. <aop:aspectj-autoproxy proxy-target-class="true"/>  
  2. <tx:annotation-driven transaction-manager="transactionManager"/>  

此配置可以很好的工作,並注入類(不是接口)。 

 

分析

1、<aop:aspectj-autoproxy proxy-target-class="true"> 該命名空間會交給org.springframework.aop.config.AopNamespaceHandler處理: 

Java代碼  收藏代碼
  1. registerBeanDefinitionParser("aspectj-autoproxy"new AspectJAutoProxyBeanDefinitionParser());  

在AspectJAutoProxyBeanDefinitionParser中,會執行parse方法解析配置:

Java代碼  收藏代碼
  1. public BeanDefinition parse(Element element, ParserContext parserContext) {  
  2.     AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);  
  3.     extendBeanDefinition(element, parserContext);  
  4.     return null;  
  5. }  

其中AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element);目的是註冊AnnotationAwareAspectJAutoProxyCreator: 

Java代碼  收藏代碼
  1. return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);  

但是注意了: 

Java代碼  收藏代碼
  1. if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {  
  2.     BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);  
  3.     if (!cls.getName().equals(apcDefinition.getBeanClassName())) {  
  4.         int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());  
  5.         int requiredPriority = findPriorityForClass(cls);  
  6.         if (currentPriority < requiredPriority) {  
  7.             apcDefinition.setBeanClassName(cls.getName());  
  8.         }  
  9.     }  
  10.     return null;  
  11. }  

大家可以看到一句話: 

  • if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) 
  • AUTO_PROXY_CREATOR_BEAN_NAME=“org.springframework.aop.config.internalAutoProxyCreator”,
  • 即首先判斷當前容器中是否包含名字爲AUTO_PROXY_CREATOR_BEAN_NAME的Bean, 如果包含:然後判斷優先級,誰優先級高誰獲勝,即最後那個獲勝的是實際的AutoProxyCreator

到此我們可以看到跟"<bean class="org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator">"配置沒什麼區別,除了沒有名字外。

 

2、接下來看一下<tx:annotation-driven>:

該命名空間交給org.springframework.transaction.config.TxNamespaceHandler處理: 

Java代碼  收藏代碼
  1. registerBeanDefinitionParser("annotation-driven"new AnnotationDrivenBeanDefinitionParser());  

其中<annotation-driven> 會交給AnnotationDrivenBeanDefinitionParser進行解析: 

Java代碼  收藏代碼
  1. public BeanDefinition parse(Element element, ParserContext parserContext) {  
  2.     String mode = element.getAttribute("mode");  
  3.     if ("aspectj".equals(mode)) {  
  4.         // mode="aspectj"  
  5.         registerTransactionAspect(element, parserContext);  
  6.     }  
  7.     else {  
  8.         // mode="proxy"  
  9.         AopAutoProxyConfigurer.configureAutoProxyCreator(element, parserContext);  
  10.     }  
  11.     return null;  
  12. }  

默認mode="proxy",所以走AopAutoProxyConfigurer.configureAutoProxyCreator,其代碼中第一句話是: 

Java代碼  收藏代碼
  1. AopNamespaceUtils.registerAutoProxyCreatorIfNecessary(parserContext, element);  
Java代碼  收藏代碼
  1. public static void registerAutoProxyCreatorIfNecessary(  
  2.             ParserContext parserContext, Element sourceElement) {  
  3.   
  4.         BeanDefinition beanDefinition = AopConfigUtils.registerAutoProxyCreatorIfNecessary(  
  5.                 parserContext.getRegistry(), parserContext.extractSource(sourceElement));  
  6.         useClassProxyingIfNecessary(parserContext.getRegistry(), sourceElement);  
  7.         registerComponentIfNecessary(beanDefinition, parserContext);  
  8.     }  

AopConfigUtils.registerAutoProxyCreatorIfNecessary是: 

Java代碼  收藏代碼
  1. registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);  
Java代碼  收藏代碼
  1. private static BeanDefinition registerOrEscalateApcAsRequired(Class cls, BeanDefinitionRegistry registry, Object source) {  
  2.         Assert.notNull(registry, "BeanDefinitionRegistry must not be null");  
  3.         if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) {  
  4.             BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME);  
  5.             if (!cls.getName().equals(apcDefinition.getBeanClassName())) {  
  6.                 int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName());  
  7.                 int requiredPriority = findPriorityForClass(cls);  
  8.                 if (currentPriority < requiredPriority) {  
  9.                     apcDefinition.setBeanClassName(cls.getName());  
  10.                 }  
  11.             }  
  12.             return null;  
  13.         }  
  14. //省略  

此處我們又看到了registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME),如果是:

  • 配置1,那麼實際是兩個AutoProxyCreator;
  • 配置2,那麼實際是共用一個AutoProxyCreator;

而且如果配置1時,因爲我們沒有指定<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/> 所以是JDK動態代理,因此不管怎麼樣,都無法注入類的。

 

問題找到了,原因是註冊了兩個AutoProxyCreator,造成了二次代理引發的問題,這個和之前的《spring的二次代理原因及如何排查》一樣。

 

如果解決

  • 給配置1起名字爲”org.springframework.aop.config.internalAutoProxyCreator“;
  • 或者使用配置2 

建議

1、如果沒有必要,請不要使用低級別API,如上述-->自己去創建AutoProxyCreator

2、首先選擇使用如:

<aop:config>

<org.springframework.aop.config.internalAutoProxyCreator>

 

如上配置已經非常好了,根本沒必要使用低級別API。

 

如<tx:annotation-driven>使用的AutoProxyCreator都是和上邊是一樣的。這樣還能防止二次代理。

 

聲明式/@AspectJ風格的AOP都非常好了,完全沒必要使用低級別API,請不要再使用低級別API了。

 

 

如果用過shiro的朋友都應該知道如下配置:

Java代碼  收藏代碼
  1. <!-- Enable Shiro Annotations for Spring-configured beans.  Only run after -->  
  2. <!-- the lifecycleBeanProcessor has run: -->  
  3. <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" depends-on="lifecycleBeanPostProcessor"/>  
  4. <bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">  
  5.     <property name="securityManager" ref="securityManager"/>  
  6. </bean>  

 

其實我們可以這樣:

Java代碼  收藏代碼
  1. <aop:config proxy-target-class="true"/>  
  2. <bean class="org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor">  
  3.     <property name="securityManager" ref="securityManager"/>  
  4. </bean>  

或者使用<aop:aspectj-autoproxy>也行,這樣也不會存在二次代理的問題。  

 

發佈了34 篇原創文章 · 獲贊 6 · 訪問量 18萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章