SPRing的AOP命名空間提供了<aop:config>、<aop:aspectj-autoproxy>和<aop:scoped-proxy>三大標(biāo)簽。前兩個(gè)標(biāo)簽的作用是想指定pointcut(切點(diǎn))提供adivce(增強(qiáng))處理,而<aop:scoped-proxy>的作用是對(duì)生命周期短的bean提供裝飾,使生命周期短的bean能被生命周期長(zhǎng)的bean正確調(diào)用。這一篇不會(huì)探討<aop:scoped-proxy>標(biāo)簽,而<aop:config>的探討見(jiàn)解析aop命名空間之config標(biāo)簽,下面我們來(lái)探討spring如何解析<aop:aspectj-autoproxy>標(biāo)簽。
<aop:aspectj-autoproxy>的解析器是AspectJAutoProxyBeanDefinitionParser,它直接實(shí)現(xiàn)了BeanDefinitionParser接口的parse方法,下面是這個(gè)方法的源碼。
@Override public BeanDefinition parse(Element element, ParserContext parserContext) { AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element); extendBeanDefinition(element, parserContext); return null; }parse方法就兩行代碼,它們分別是想容器中注冊(cè)自動(dòng)代理構(gòu)建器對(duì)象和解析<aop:aspectj-autoproxy>的子標(biāo)簽<aop:include>。我們分別來(lái)看看這兩件事情的細(xì)節(jié)。
1 創(chuàng)建自動(dòng)代理構(gòu)建器對(duì)象 parse方法調(diào)用AOP命名空間工具類(lèi)AopNamespaceUtils的registerAspectJAnnotationAutoProxyCreatorIfNecessary方法來(lái)完成向容器中注冊(cè)自動(dòng)代理構(gòu)建器對(duì)象的任務(wù),源碼如下。
public static void registerAspectJAnnotationAutoProxyCreatorIfNecessary( ParserContext parserContext, Element sourceElement) { // 注冊(cè)自動(dòng)代理構(gòu)建器對(duì)象 BeanDefinition beanDefinition = AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary( parserContext.getRegistry(), parserContext.extractSource(sourceElement)); // 解析proxy-target-class和expose-proxy屬性 useClassProxyingIfNecessary(parserContext.getRegistry(), sourceElement); registerComponentIfNecessary(beanDefinition, parserContext); }繼續(xù)看AOP配置工具類(lèi)AopConfigUtils的registerAspectJAnnotationAutoProxyCreatorIfNecessary方法,源碼如下。
public static BeanDefinition registerAspectJAnnotationAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) { // 向容器中注冊(cè)一個(gè)AnnotationAwareAspectJAutoProxyCreator對(duì)象 return registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source); }下面是AopConfigUtils工具類(lèi)的registerOrEscalateApcAsRequired方法
private static BeanDefinition registerOrEscalateApcAsRequired(Class<?> cls, BeanDefinitionRegistry registry, Object source) { Assert.notNull(registry, "BeanDefinitionRegistry must not be null"); // 定義有AUTO_PROXY_CREATOR_BEAN_NAME="org.springframework.aop.config.internalAutoProxyCreator" if (registry.containsBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME)) { // 如果容器中已經(jīng)存在自動(dòng)代理構(gòu)建器,則比較兩個(gè)構(gòu)建器的優(yōu)先級(jí) BeanDefinition apcDefinition = registry.getBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME); if (!cls.getName().equals(apcDefinition.getBeanClassName())) { int currentPriority = findPriorityForClass(apcDefinition.getBeanClassName()); int requiredPriority = findPriorityForClass(cls); // 保存優(yōu)先級(jí)高的構(gòu)建器 if (currentPriority < requiredPriority) { apcDefinition.setBeanClassName(cls.getName()); } } return null; } // 如果容器中還沒(méi)有自動(dòng)代理構(gòu)建器 // 則創(chuàng)建構(gòu)建器相應(yīng)的BeanDefinition對(duì)象 RootBeanDefinition beanDefinition = new RootBeanDefinition(cls); beanDefinition.setSource(source); beanDefinition.getPropertyValues().add("order", Ordered.HIGHEST_PRECEDENCE); beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE); // 向容器中注冊(cè)代理構(gòu)建器的BeanDefinition對(duì)象 registry.registerBeanDefinition(AUTO_PROXY_CREATOR_BEAN_NAME, beanDefinition); return beanDefinition; }AopConfigUtils獲取代理構(gòu)建器的優(yōu)先級(jí)方法內(nèi)容如下。
private static int findPriorityForClass(Class<?> clazz) { return APC_PRIORITY_LIST.indexOf(clazz); }其中AopConfigUtils中定義了如下自動(dòng)代理構(gòu)建器
private static final List<Class<?>> APC_PRIORITY_LIST = new ArrayList<Class<?>>(); static { APC_PRIORITY_LIST.add(InfrastructureAdvisorAutoProxyCreator.class); APC_PRIORITY_LIST.add(AspectJAwareAdvisorAutoProxyCreator.class); APC_PRIORITY_LIST.add(AnnotationAwareAspectJAutoProxyCreator.class); }從上面代碼可以看出AnnotationAwareAspectJAutoProxyCreator的優(yōu)先級(jí)最高,其次是AspectJAwareAdvisorAutoProxyCreator,而InfrastructureAdvisorAutoProxyCreator的優(yōu)先級(jí)最低。
2 解析<aop:include>子標(biāo)簽
AspectJAutoProxyBeanDefinitionParser的parse方法調(diào)用這個(gè)類(lèi)的extendBeanDefinition方法來(lái)完成解析<aop:include>標(biāo)簽的解析,這個(gè)方法的源碼如下。
private void extendBeanDefinition(Element element, ParserContext parserContext) { // 獲得自動(dòng)代理構(gòu)建器的BeanDefinition BeanDefinition beanDef = parserContext.getRegistry().getBeanDefinition(AopConfigUtils.AUTO_PROXY_CREATOR_BEAN_NAME); if (element.hasChildNodes()) { addIncludePatterns(element, parserContext, beanDef); } }extendBeanDefinition方法的作用是獲取自動(dòng)代理構(gòu)建器的BeanDefinition對(duì)象和判斷配置文件中<aop:aspectj-autoproxy>的定義是否還有子節(jié)點(diǎn)。下面繼續(xù)看AspectJAutoProxyBeanDefinitionParser的addIncludePatterns方法,源碼如下。
private void addIncludePatterns(Element element, ParserContext parserContext, BeanDefinition beanDef) { ManagedList<TypedStringValue> includePatterns = new ManagedList<TypedStringValue>(); NodeList childNodes = element.getChildNodes(); // 遍歷子節(jié)點(diǎn) for (int i = 0; i < childNodes.getLength(); i++) { Node node = childNodes.item(i); // apsect-autoproxy標(biāo)簽只有一個(gè)子標(biāo)簽,那就是include標(biāo)簽 // 這里判斷節(jié)點(diǎn)是否為Element對(duì)象 if (node instanceof Element) { Element includeElement = (Element) node; // 獲取include標(biāo)簽的name屬性 TypedStringValue valueHolder = new TypedStringValue(includeElement.getAttribute("name")); valueHolder.setSource(parserContext.extractSource(includeElement)); includePatterns.add(valueHolder); } } if (!includePatterns.isEmpty()) { includePatterns.setSource(parserContext.extractSource(element)); // 使用解析inculde標(biāo)簽獲得的includePatterns來(lái)設(shè)置代理構(gòu)建器的includePatterns屬性。 beanDef.getPropertyValues().add("includePatterns", includePatterns); } }<aop:aspectj-autoproxy>和<aop:config>一樣,首先會(huì)向spring容器中注冊(cè)自動(dòng)代理構(gòu)建器。<aop:config>注冊(cè)的是AspectJAwareAdvisorAutoProxyCreator對(duì)象,它用于暴露AspectJ的執(zhí)行上下文和遵從AspectJ判讀來(lái)自同一個(gè)aspect(切面)的多個(gè)advice(增強(qiáng))的執(zhí)行順序的規(guī)則。<aop:aspectj-autoproxy>注冊(cè)的是AnnotationAwareAspectJAutoProxyCreator對(duì)象,它繼承了AspectJAwareAdvisorAutoProxyCreator類(lèi),并處理當(dāng)前所處spring容器中所有的AspectJ注解。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注