Spring4.3.x 浅析xml配置的解析过程(11)——解析aop命名空间之scoped-proxy标签
来源:互联网 发布:mac导入图片到iphone 编辑:程序博客网 时间:2024/05/18 20:13
概述
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>标签
<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标签一节。
- Spring4.3.x 浅析xml配置的解析过程(11)——解析aop命名空间之scoped-proxy标签
- Spring4.3.x 浅析xml配置的解析过程(9)——解析aop命名空间之config标签
- Spring4.3.x 浅析xml配置的解析过程(10)——解析aop命名空间之aspectj-autoproxy标签
- Spring4.3.x 浅析xml配置的解析过程(5)——解析自定义命名空间的标签
- Spring4.3.x 浅析xml配置的解析过程(6)——解析context命名空间之property-placeholder和property-override标签
- Spring4.3.x 浅析xml配置的解析过程(7)——解析context命名空间之annotation-config标签
- Spring4.3.x 浅析xml配置的解析过程(8)——解析context命名空间之component-scan标签
- Spring4.3.x 浅析xml配置的解析过程(4)——解析bean标签及其所有子标签
- Spring4.3.x 浅析xml配置的解析过程(1)——使用XmlBeanDefinitionReader解析xml配置
- Spring4.3.x 浅析xml配置的解析过程(2)——使用ResourceLoader创建Resource对象
- Spring4.3.x 浅析xml配置的解析过程(3)——使用DocumentLoader创建Document对象
- Spring学习笔记 —— AOP标签详解(<scoped-proxy>)
- 在Spring的XML文件中通过命名空间配置的标签是怎么解析的?
- XML命名空间深度解析 —— 命名空间简介
- xml 解析 命名空间
- aop xml 配置方式解析过程
- dom4j 解析带命名空间的XML
- PHP 解析有命名空间的 XML
- MVC,MVP,MVVM的区别
- linux查看系统信息命令
- Java HashSet和HashMap源码剖析
- [Android新手学习笔记01]-如何手动创建Activity
- html5多媒体标签之audio标签
- Spring4.3.x 浅析xml配置的解析过程(11)——解析aop命名空间之scoped-proxy标签
- HTTP访问控制(CORS) 跨域访问
- ubuntu下安装opencv3.2出现-- ICV: Downloading ippicv_linux_20151201.tgz...
- 重拾算法之剑指Offier——调整数组顺序使奇数位于偶数前面
- struts2的核心和工作原理
- const的常见用法
- 350. Intersection of Two Arrays II
- Mysql5.5的二进制安装
- [Android新手学习笔记02]-如何创建Button并实现点击事件