SPRing為生命周期長的bean調用生命周期短的bean提供了三種解決方案。第一種是使用默認命名空間(beans)的<look-up>標簽;第二種是使用context命名空間的<context:component-scan>解析@Scope注解;第三種是使用AOP命名空間的<aop:scoped-proxy>標簽裝飾生命周期短的bean。<aop:scoped-proxy>的使用如下
<bean id="user" class="com.chyohn.User" scope="session"> <aop:scoped-proxy/></bean><bean id="userManager" class="com.chyohn.UserManager" scope="singleton"> <property name="targetUser" ref="user"/></bean><aop:scoped-proxy>是AOP命名空間的三大標簽之一,它的作用是對生命周期短的bean提供裝飾,使其能被生命周期長的bean正確調用,下面我們來探討Spring是如何解析<aop:scoped-proxy>標簽的。
<aop:scoped-proxy>標簽屬于spring<bean>標簽的裝飾標簽,它的裝飾器是ScopedProxyBeanDefinitionDecorator,它直接繼承了BeanDefinitionDecorator接口的decorate方法,這個方法的源碼如下。
@Override public BeanDefinitionHolder decorate(Node node, BeanDefinitionHolder definition, ParserContext parserContext) { boolean proxyTargetClass = true; if (node instanceof Element) { Element ele = (Element) node; if (ele.hasAttribute("proxy-target-class")) { // 設置是用CGLIB還是JDK動態代理,true使用前者,false使用后者。默認為true,即使用CGLIB proxyTargetClass = Boolean.valueOf(ele.getAttribute("proxy-target-class")); } } // 調用作用域代理工具類ScopedProxyUtils創建作用域代理 // 注冊被裝飾的BeanDefinition,并返回代理BeanDefintion BeanDefinitionHolder holder = ScopedProxyUtils.createScopedProxy(definition, parserContext.getRegistry(), proxyTargetClass); String targetBeanName = ScopedProxyUtils.getTargetBeanName(definition.getBeanName()); parserContext.getReaderContext().fireComponentRegistered( new BeanComponentDefinition(definition.getBeanDefinition(), targetBeanName)); return holder; }繼續看作用域代理工具類ScopedProxyUtils的createScopedProxy方法源碼如下。
public static BeanDefinitionHolder createScopedProxy(BeanDefinitionHolder definition, BeanDefinitionRegistry registry, boolean proxyTargetClass) { String originalBeanName = definition.getBeanName(); BeanDefinition targetDefinition = definition.getBeanDefinition(); // targetBeanName格式為scopedTarget. + originalBeanName String targetBeanName = getTargetBeanName(originalBeanName); // Create a scoped proxy definition for the original bean name, // "hiding" the target bean in an internal target definition. // 創建一個ScopedProxyFactoryBean類對應BeanDefinition對象 RootBeanDefinition proxyDefinition = new RootBeanDefinition(ScopedProxyFactoryBean.class); proxyDefinition.setDecoratedDefinition(new BeanDefinitionHolder(targetDefinition, targetBeanName)); proxyDefinition.setOriginatingBeanDefinition(targetDefinition); proxyDefinition.setSource(definition.getSource()); proxyDefinition.setRole(targetDefinition.getRole()); proxyDefinition.getPropertyValues().add("targetBeanName", targetBeanName); if (proxyTargetClass) { targetDefinition.setAttribute(AutoProxyUtils.PRESERVE_TARGET_CLASS_ATTRIBUTE, Boolean.TRUE); } else { // 設置為根據接口做做代理 proxyDefinition.getPropertyValues().add("proxyTargetClass", Boolean.FALSE); } // 根據代理目標BeanDefinition設置是否可以為自動注入的候選bean proxyDefinition.setAutowireCandidate(targetDefinition.isAutowireCandidate()); proxyDefinition.setPrimary(targetDefinition.isPrimary()); if (targetDefinition instanceof AbstractBeanDefinition) { proxyDefinition.copyQualifiersFrom((AbstractBeanDefinition) targetDefinition); } // 隱藏被代理的bean targetDefinition.setAutowireCandidate(false); targetDefinition.setPrimary(false); // 注冊被代理的bean的BeanDefinition對象 registry.registerBeanDefinition(targetBeanName, targetDefinition); // 返回代理bean的BeanDefinitionHolder對象 return new BeanDefinitionHolder(proxyDefinition, originalBeanName, definition.getAliases()); }createScopedProxy方法向容器中創建了ScopedProxyFactoryBean對象用于代理bean。
<aop:scoped-proxy>的作用域代理方式和@Scope注解的代理方式一樣,都是通過ScopedProxyFactoryBean對象來代理的。兩者的不同在于一個是基于xml配置,一個是基于注解配置的。
關于@Scope注解的解析見解析context命名空間之component-scan標簽中關于解析component-scan標簽一節。
新聞熱點
疑難解答