美文网首页
第3章 默认标签的解析

第3章 默认标签的解析

作者: 十丈_红尘 | 来源:发表于2018-12-31 23:34 被阅读0次

      之前提到过Spring中的标签包括默认标签和自定义标签两种,而两种标签的用法以及解析方式存在着很大的不同,本章节重点分析默认标签的解析过程.
      默认标签的解析是在parseDefaultElement方法中进行的,方法中的功能逻辑一目了然,分别对4种不同标签(import alias beanbeans)做了不同的处理.

    private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {
            // 对import标签处理 
            if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) {
                importBeanDefinitionResource(ele);
            }
            // 对alias标签处理
            else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) {
                processAliasRegistration(ele);
            }
            // 对bean标签处理
            else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) {
                processBeanDefinition(ele, delegate);
            }
            // 对beans标签处理
            else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) {
                // recurse
                doRegisterBeanDefinitions(ele);
            }
        }
    

    3.1 bean标签的解析及注册

      在4种标签的解析中,对bean标签的解析最为复杂也最为重要,所以我们从此标签开始深入分析,如果能理解此标签的解析过程,其他标签的解析自然会迎刃而解.首先我们进入方法processBeanDefinition(ele, delegate)中.

    protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
            BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
            if (bdHolder != null) {
                bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
                try {
                    // Register the final decorated instance.
                    BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
                }
                catch (BeanDefinitionStoreException ex) {
                    getReaderContext().error("Failed to register bean definition with name '" +
                            bdHolder.getBeanName() + "'", ele, ex);
                }
                // Send registration event.
                getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
            }
        }
    

    此方法大致逻辑如下:

    1. 首先委托BeanDefinitionParserDelegate类的 parseBeanDefinitionElement方法进行元素解析,返回BeanDefinitionHolder类型的实例bdHolder,经过这个方法后bdHolder实例已经包含我们配置文件中配置的各种属性了,例如class name id alias之类的属性.
    2. 当返回bdHolder不为空的情况下若存在默认标签的子节点下再有自定义属性,还需要再次对自定义标签进行解析.
    3. 解析完成后,需要对解析后的bdHolder进行注册,同样,注册操作委托给了BeanDefinitionReaderUtilsregisterBeanDefinition方法.
    4. 最后发出响应事件,通知相关的监听器,这个bean已经加载完成了.
    时序图(图 3-1)如下: 图 3-1

    3.1.1 解析BeanDefinition

      下面我们就针对各个操作做具体分析.首先我们从元素解析及信息提取开始,也就是BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele),进入BeanDefinitionParserDelegate类的parseBeanDefinitionElement方法.

    @Nullable
        public BeanDefinitionHolder parseBeanDefinitionElement(Element ele) {
            return parseBeanDefinitionElement(ele, null);
        }
    
    @Nullable
        public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean) {
            // 解析id属性
            String id = ele.getAttribute(ID_ATTRIBUTE);
            // 解析name属性
            String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);
            // 分割name属性
            List<String> aliases = new ArrayList<>();
            if (StringUtils.hasLength(nameAttr)) {
                String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
                aliases.addAll(Arrays.asList(nameArr));
            }
    
            String beanName = id;
            if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
                beanName = aliases.remove(0);
                if (logger.isDebugEnabled()) {
                    logger.debug("No XML 'id' specified - using '" + beanName +
                            "' as bean name and " + aliases + " as aliases");
                }
            }
    
            if (containingBean == null) {
                checkNameUniqueness(beanName, aliases, ele);
            }
    
            AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);
            if (beanDefinition != null) {
                if (!StringUtils.hasText(beanName)) {
                    try {
                        // 如果不存在beanName那么根据Spring中提供的命名规则为当前bean生成对应的beanName
                        if (containingBean != null) {
                            beanName = BeanDefinitionReaderUtils.generateBeanName(
                                    beanDefinition, this.readerContext.getRegistry(), true);
                        }
                        else {
                            beanName = this.readerContext.generateBeanName(beanDefinition);
                            // Register an alias for the plain bean class name, if still possible,
                            // if the generator returned the class name plus a suffix.
                            // This is expected for Spring 1.2/2.0 backwards compatibility.
                            String beanClassName = beanDefinition.getBeanClassName();
                            if (beanClassName != null &&
                                    beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&
                                    !this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {
                                aliases.add(beanClassName);
                            }
                        }
                        if (logger.isDebugEnabled()) {
                            logger.debug("Neither XML 'id' nor 'name' specified - " +
                                    "using generated bean name [" + beanName + "]");
                        }
                    }
                    catch (Exception ex) {
                        error(ex.getMessage(), ele);
                        return null;
                    }
                }
                String[] aliasesArray = StringUtils.toStringArray(aliases);
                return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
            }
    
            return null;
        }
    

      以上便是对默认标签解析的全过程了.Spring的解析犹如剥洋葱一样一层一层地进行,尽管现在只能看到对属性id以及name的解析,但是很庆幸,思路我们已经了解了.在开始对属性展开全面解析前,Spring在外层又做了一个当前层的功能架构,在当前层完成的主要工作包括如下内容.

    1. 提取元素中的id以及name属性.
    2. 进一步解析其他所有属性并统一封装至GenericBeanDefinition类型的实例中.
    3. 如果检测到bean没有指定beanName,那么使用默认规则为此Bean生成beanName.
    4. 将获取到的信息封装到BeanDefinitionHolder的实例中.

    我们进一步查看步骤2中对标签其他属性的解析过程.

        @Nullable
        public AbstractBeanDefinition parseBeanDefinitionElement(
                Element ele, String beanName, @Nullable BeanDefinition containingBean) {
    
            this.parseState.push(new BeanEntry(beanName));
    
            String className = null;
            // 解析class属性
            if (ele.hasAttribute(CLASS_ATTRIBUTE)) {
                className = ele.getAttribute(CLASS_ATTRIBUTE).trim();
            }
            String parent = null;
            // 解析parent属性
            if (ele.hasAttribute(PARENT_ATTRIBUTE)) {
                parent = ele.getAttribute(PARENT_ATTRIBUTE);
            }
    
            try {
                // 创建用于承载属性的AbstractBeanDefinition类型的createBeanDefinition
                AbstractBeanDefinition bd = createBeanDefinition(className, parent);
                // 硬编码解析默认bean的各种属性
                parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
                // 提取Description
                bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));
                // 解析元数据
                parseMetaElements(ele, bd);
                // 解析lookup-method属性
                parseLookupOverrideSubElements(ele, bd.getMethodOverrides());
                // 解析replaced-method属性
                parseReplacedMethodSubElements(ele, bd.getMethodOverrides());
                // 解析构造函数参数
                parseConstructorArgElements(ele, bd);
                // 解析property子元素
                parsePropertyElements(ele, bd);
                // 解析qualifier子元素
                parseQualifierElements(ele, bd);
    
                bd.setResource(this.readerContext.getResource());
                bd.setSource(extractSource(ele));
    
                return bd;
            }
            catch (ClassNotFoundException ex) {
                error("Bean class [" + className + "] not found", ele, ex);
            }
            catch (NoClassDefFoundError err) {
                error("Class that bean class [" + className + "] depends on not found", ele, err);
            }
            catch (Throwable ex) {
                error("Unexpected failure during bean definition parsing", ele, ex);
            }
            finally {
                this.parseState.pop();
            }
    
            return null;
        }
    

      终于,bean标签的所有属性,不论常用的还是不常用的我们都看到了,尽管有些复杂的属性还需要进一步的解析,接下来我们继续一些复杂标签的解析.


    1 创建用于属性承载的BeanDefinition

      BeanDefinition是一个接口,在Spring中存在三种实现:RootBeanDefinition ChildBeanDefinition 以及 GenericBeanDefinition.三种实现均继承了AbstractBeanDefinition,其中BeanDefinition是配置文件<bean>元素标签在容器中的内部表示形式.<bean>元素标签拥有class scope lazy-init等配置属性,BeanDefinition则提供了响应的beanClass scope lazyInit属性,BeanDefinition<bean>中的属性是一一对应的.其中RootBeanDefinition是最常用的实现类,它对应一般性的<bean>元素标签,GenericBeanDefinition是自2.5版本以后新加入的bean文件配置属性定义类,是一站式服务类.
      在配置文件中可以定义父<bean>和子<bean>,父<bean>RootBeanDefinition表示,而子<bean>ChildBeanDefinition表示,而没有父<bean><bean>就使用RootBeanDefinition表示.AbstractBeanDefinition对两者共同的类信息进行抽象.
      Spring通过BeanDefinition将配置文件中的<bean>配置信息转换为容器的内部表示,并将这些BeanDefinition注册到BeanDefinitionRegistry中.Spring容器的BeanDefinitionRegistry就像是Spring配置信息的内存数据库,主要是以map的形式保存,后续操作直接从BeanDefinitionRegistry中读取配置信息.它们之间的关系如图 3-2 所示:

    图 3-2   由此可知,要解析属性首先要创建用于承载属性的实例,也就是创建GenericBeanDefinition类型的实例.而代码createBeanDefinition(className, parent)的作用就是实现此功能.
    protected AbstractBeanDefinition createBeanDefinition(@Nullable String className, @Nullable String parentName)
                throws ClassNotFoundException {
    
            return BeanDefinitionReaderUtils.createBeanDefinition(
                    parentName, className, this.readerContext.getBeanClassLoader());
        }
    
        public static AbstractBeanDefinition createBeanDefinition(
                @Nullable String parentName, @Nullable String className, @Nullable ClassLoader classLoader) throws ClassNotFoundException {
    
            GenericBeanDefinition bd = new GenericBeanDefinition();
            // parentName可能为空
            bd.setParentName(parentName);
            if (className != null) {
                if (classLoader != null) {
                    // 如果classLoader不为空,则使用以传入的classLoader同一虚拟机类加载对象,否则只是记录className
                    bd.setBeanClass(ClassUtils.forName(className, classLoader));
                }
                else {
                    bd.setBeanClassName(className);
                }
            }
            return bd;
        }
    
    2. 解析各种属性

      当我们创建了bean信息的承载实例后,便可以进行bean信息的各种属性解析了,首先我们进入parseBeanDefinitionAttributes方法.parseBeanDefinitionAttributes方法是对element所有元素属性进行解析:

        public AbstractBeanDefinition parseBeanDefinitionAttributes(Element ele, String beanName,
                @Nullable BeanDefinition containingBean, AbstractBeanDefinition bd) {
            // 解析singleton属性
            if (ele.hasAttribute(SINGLETON_ATTRIBUTE)) {
                // scope和singleton两个属性只能指定其中一个,不可以同时出现,否则Spring将会抛出异常
                error("Old 1.x 'singleton' attribute in use - upgrade to 'scope' declaration", ele);
            }
            // 解析scope属性
            else if (ele.hasAttribute(SCOPE_ATTRIBUTE)) {
                bd.setScope(ele.getAttribute(SCOPE_ATTRIBUTE));
            }
    
            else if (containingBean != null) {
                // 在嵌入beanDifinition情况下没有单独指定scope属性则使用父类默认的属性
                bd.setScope(containingBean.getScope());
            }
            // 解析abstract属性
            if (ele.hasAttribute(ABSTRACT_ATTRIBUTE)) {
                bd.setAbstract(TRUE_VALUE.equals(ele.getAttribute(ABSTRACT_ATTRIBUTE)));
            }
    
            String lazyInit = ele.getAttribute(LAZY_INIT_ATTRIBUTE);
            // 解析lazy-init属性
            if (DEFAULT_VALUE.equals(lazyInit)) {
                lazyInit = this.defaults.getLazyInit();
            }
            // 若没有设置或设置成其他字符都会被设置为false
            bd.setLazyInit(TRUE_VALUE.equals(lazyInit));
            // 解析autowire属性
            String autowire = ele.getAttribute(AUTOWIRE_ATTRIBUTE);
            bd.setAutowireMode(getAutowireMode(autowire));
            // 解析depends-on属性
            if (ele.hasAttribute(DEPENDS_ON_ATTRIBUTE)) {
                String dependsOn = ele.getAttribute(DEPENDS_ON_ATTRIBUTE);
                bd.setDependsOn(StringUtils.tokenizeToStringArray(dependsOn, MULTI_VALUE_ATTRIBUTE_DELIMITERS));
            }
            // 解析autowire-candidate属性
            String autowireCandidate = ele.getAttribute(AUTOWIRE_CANDIDATE_ATTRIBUTE);
            if ("".equals(autowireCandidate) || DEFAULT_VALUE.equals(autowireCandidate)) {
                String candidatePattern = this.defaults.getAutowireCandidates();
                if (candidatePattern != null) {
                    String[] patterns = StringUtils.commaDelimitedListToStringArray(candidatePattern);
                    bd.setAutowireCandidate(PatternMatchUtils.simpleMatch(patterns, beanName));
                }
            }
            else {
                bd.setAutowireCandidate(TRUE_VALUE.equals(autowireCandidate));
            }
            // 解析primary属性
            if (ele.hasAttribute(PRIMARY_ATTRIBUTE)) {
                bd.setPrimary(TRUE_VALUE.equals(ele.getAttribute(PRIMARY_ATTRIBUTE)));
            }
            // 解析init-method属性
            if (ele.hasAttribute(INIT_METHOD_ATTRIBUTE)) {
                String initMethodName = ele.getAttribute(INIT_METHOD_ATTRIBUTE);
                bd.setInitMethodName(initMethodName);
            }
            else if (this.defaults.getInitMethod() != null) {
                bd.setInitMethodName(this.defaults.getInitMethod());
                bd.setEnforceInitMethod(false);
            }
            // 解析destroy-method属性
            if (ele.hasAttribute(DESTROY_METHOD_ATTRIBUTE)) {
                String destroyMethodName = ele.getAttribute(DESTROY_METHOD_ATTRIBUTE);
                bd.setDestroyMethodName(destroyMethodName);
            }
            else if (this.defaults.getDestroyMethod() != null) {
                bd.setDestroyMethodName(this.defaults.getDestroyMethod());
                bd.setEnforceDestroyMethod(false);
            }
            // 解析factory-method属性
            if (ele.hasAttribute(FACTORY_METHOD_ATTRIBUTE)) {
                bd.setFactoryMethodName(ele.getAttribute(FACTORY_METHOD_ATTRIBUTE));
            }
            // 解析factory-bean属性
            if (ele.hasAttribute(FACTORY_BEAN_ATTRIBUTE)) {
                bd.setFactoryBeanName(ele.getAttribute(FACTORY_BEAN_ATTRIBUTE));
            }
    
            return bd;
        }
    
    3. 解析子元素meta

      在开始解析元数据的分析前,我们先回顾下元数据meta属性的使用.

    <bean id="myTestBean" class="bean.MyTestBean">
          <meta key="testStr" value="aaaaaaaa"/>
    </bean>
    

      这段代码并不会体现在MyTestBean的属性当中,而是一个额外的声明,当需要使用里面的信息的时候可以通过BeanDefinitiongetAttribute(key)方法进行获取.对meta属性的解析如下:

    public void parseMetaElements(Element ele, BeanMetadataAttributeAccessor attributeAccessor) {
            // 获取当前节点的所有子元素
            NodeList nl = ele.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                // 提取meta
                if (isCandidateElement(node) && nodeNameEquals(node, META_ELEMENT)) {
                    Element metaElement = (Element) node;
                    String key = metaElement.getAttribute(KEY_ATTRIBUTE);
                    String value = metaElement.getAttribute(VALUE_ATTRIBUTE);
                    // 使用key value构造BeanMetadataAttribute
                    BeanMetadataAttribute attribute = new BeanMetadataAttribute(key, value);
                    attribute.setSource(extractSource(metaElement));
                    // 记录信息
                    attributeAccessor.addMetadataAttribute(attribute);
                }
            }
        }
    
    4. 解析子元素lookup-method

      同样,子元素lookup-method似乎并不是很常用,但是在某些时候它的确是非常有用的属性,通常我们称它为获取器注入.引用<Spring in action>中的一句话:获取器注入是一种特殊的方法注入,它是把一个方法声明为返回某种类型的bean,但实际要返回的bean是在配置文件里面配置的,此方法可用在设计有些可插拔的功能上,接触程序依赖.我们看看具体的应用.

    1. 首先我们创建一个父类
    package.test.lookup.bean
    
    public class User {
        public void showMe(){
            System.out.println("i am user");
        }
    }
    
    1. 创建其子类并覆盖showMe方法.
    package.test.lookup.bean
    
    public class Teacher extends User {
        public void showMe(){
            System.out.println("i am Teacher");
        }
    }
    
    1. 创建调用方法.
    public abstract class GetBeanTest {
          public void showMe(){
                this.getBean().showMe();
          }
          public abstract User getBean();
    }
    
    1. 创建测试方法
    package test.lookup
    
    import org.Springframework.context.ApplicationContext;
    import org.Springframework.context.support.ClassPathXmlApplicationContext;
    import test.lookup.app.GetBeanTest;
    
    public class Main {
        public static void main(String[] args){
            ApplicationContext bf = new ClassPathXmlApplicationContext("test/lookup/lookupTest.xml");
            GetBeanTest test = (GetBeanTest) bf.getBean("GetBeanTest");
            test.showMe();
        }
    }
    

      到现在为止,除了配置文件外,整个测试方法就完成了,如果之前没有接触过获取器注入的同学可能会有疑问:抽象方法还没有被实现,怎么可以直接调用呢?答案就在Spring为我们提供的获取器中,我们看看配置文件是怎么配置的.

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="getBeanTest" class="test.lookup.app.GetBeanTest">
            <lookup-method name="getBean" bean="teacher" />
        </bean>
    </beans>
    

      在配置文件中,我们看到了源码解析中提到的lookup-method子元素,这个配置完成的功能是动态地将teacher所代表的bean作为getBean的返回值,运行测试方法我们会看到控制台上的输出:i am Teacher
      当我们的业务变更或者在其他情况下,teacher里面的业务逻辑已经不再符合我们的业务要求,需要进行替换怎么办呢?以下是我们需要增加的新的逻辑类:

    package test.lookup.bean;
    
    public class Student extends User{
        public void showMe(){
            System.out.println("i am Student");
        }
    }
    

    同时修改配置文件:

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="getBeanTest" class="test.lookup.app.GetBeanTest">
            <lookup-method name="getBean" bean="student"/>
        </bean>
    
        <bean id="teacher" class="test.lookup.bean.Teacher"/>
        <bean id="student" class="test.lookup.bean.Student"/>
    </beans>
    

      再次运行测试类,你会发现不一样的结果:i am Student
      至此,我们已经初步了解了lookup-method子元素所提供的大致功能,这时候再去看源码应该会更有针对性了:

        public void parseLookupOverrideSubElements(Element beanEle, MethodOverrides overrides) {
            NodeList nl = beanEle.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                // 仅当在Spring默认bean的子元素下且为 < lookup-method时有效
                if (isCandidateElement(node) && nodeNameEquals(node, LOOKUP_METHOD_ELEMENT)) {
                    Element ele = (Element) node;
                    // 获取要修改的方法
                    String methodName = ele.getAttribute(NAME_ATTRIBUTE);
                    // 获取配置返回的bean
                    String beanRef = ele.getAttribute(BEAN_ELEMENT);
                    LookupOverride override = new LookupOverride(methodName, beanRef);
                    override.setSource(extractSource(ele));
                    overrides.addOverride(override);
                }
            }
        }
    

      以上的代码很眼熟,似乎与parseMetaElements的代码大同小异,最大的区别就是在if判断中的节点名称在这里被修改为LOOKUP_METHOD_ELEMENT.还有,在数据存储上面通过使用LookupOverride类型的实体类来进行数据承载并记录在AbstractBeanDefinition中的methodOverrides属性中.

    5. 解析子元素replaced-method

      这个方法主要是对beanreplaced-method子元素的提取,在开始提取分析之前我们还是预先介绍下这个元素的用法.
      方法替换 : 可以在运行时用新的方法替换现有的方法.与之前的look-up不同的是,replaced-method不但可以动态的替换返回实体bean,而且还能动态地更改原有方法的逻辑.我们来看看使用示例.
    (1) 在changeMe中完成某个业务逻辑

    public class TestChangeMethod {
        public void changeMe(){
            System.out.println("changeMe");
        }
    }
    

    (2) 在运行了一段时间后需要改变原有的业务逻辑

    public class TestMethodReplacer implements MethodReplacer{
        @Override
        public Object reimplement(Object obj, Method method, Object[] args)throws Throwable{
            System.out.println("我替换了原有的方法");
            return null;
        }
    }
    

    (3) 使替换后的类生效

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="testChangeMethod" class="test.Replacemethod.TestChangeMethod">
            <replaced-method name="changeMe" replacer="replacer"/>
        </bean>
    
        <bean id="replacer" class="test.replacemethod.TestMethodReplacer"/>
    </beans>
    

    (4) 测试

    public class test {
    
        public static void main(String[] args) {
            ClassPathXmlApplicationContext bf = new ClassPathXmlApplicationContext("test/replacemethod/replaceMethodTest.xml");
            TestChangeMethod test = (TestChangeMethod) bf.getBean("testChangeMethod");
            test.changeMe();
        }
    }
    

      好了,运行测试类就可以看到预期的结果了,控制台成功打印出"我替换了原有的方法",也就是说我们做到了动态替换原有方法,知道了这个元素的用法,我们再次来看元素的提取过程:

    public void parseReplacedMethodSubElements(Element beanEle, MethodOverrides overrides) {
            NodeList nl = beanEle.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                // 仅当在Spring默认bean的子元素下且为<replaced-method>时有效
                if (isCandidateElement(node) && nodeNameEquals(node, REPLACED_METHOD_ELEMENT)) {
                    Element replacedMethodEle = (Element) node;
                    // 提取要替换的旧方法
                    String name = replacedMethodEle.getAttribute(NAME_ATTRIBUTE);
                    // 提取对应的新的替换方法
                    String callback = replacedMethodEle.getAttribute(REPLACER_ATTRIBUTE);
                    ReplaceOverride replaceOverride = new ReplaceOverride(name, callback);
                    // Look for arg-type match elements.
                    List<Element> argTypeEles = DomUtils.getChildElementsByTagName(replacedMethodEle, ARG_TYPE_ELEMENT);
                    for (Element argTypeEle : argTypeEles) {
                        // 记录参数
                        String match = argTypeEle.getAttribute(ARG_TYPE_MATCH_ATTRIBUTE);
                        match = (StringUtils.hasText(match) ? match : DomUtils.getTextValue(argTypeEle));
                        if (StringUtils.hasText(match)) {
                            replaceOverride.addTypeIdentifier(match);
                        }
                    }
                    replaceOverride.setSource(extractSource(replacedMethodEle));
                    overrides.addOverride(replaceOverride);
                }
            }
        }
    

      我们可以看到无论是look-up还是replaced-method都是构造了一个MethodOverride,并最终记录在了AbstractBeanDefinition中的methodOverrides属性中.关于这个属性如何使用我们后边在进行介绍.

    6. 解析子元素construction-arg

      对构造函数的解析是非常常用的,同时也是非常复杂的,也相信大家对构造函数的配置都不陌生,举个简单的小例子:

    <beans>
        <!-- 默认的情况下是按照参数的顺序注入,当指定index索引后就可以改变注入参数的顺序 -->
        <bean id="helloBean" class="com.HelloBean">
            <constructor-arg index="0">
                <value>Spring</value>
            </constructor-arg>
            <constructor-arg index="1">
                <value>你好</value>
            </constructor-arg>
        </bean>
    </beans>
    

      上面的配置是Spring构造函数配置中最基础的配置,实现的功能就是对HelloBean自动寻找对应的构造函数,并在初始化的时候将设置的参数传入进去.那么让我们来看看具体的XML解析过程.
      对于constructor-arg子元素的解析,Spring是通过parseConstructorArgElements方法来实现的,具体的代码如下:

    public void parseConstructorArgElements(Element beanEle, BeanDefinition bd) {
            NodeList nl = beanEle.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                if (isCandidateElement(node) && nodeNameEquals(node, CONSTRUCTOR_ARG_ELEMENT)) {
                    // 解析constructor-arg
                    parseConstructorArgElement((Element) node, bd);
                }
            }
        }
    

      这个结构似乎我们可以想象得到,遍历所有子元素,也就是提取所有constructor-arg,然后进行解析,但是具体的解析却被放置在了另一个方法parseConstructorArgElement,具体代码如下:

    public void parseConstructorArgElement(Element ele, BeanDefinition bd) {
            // 提取index属性
            String indexAttr = ele.getAttribute(INDEX_ATTRIBUTE);
            // 提取type属性
            String typeAttr = ele.getAttribute(TYPE_ATTRIBUTE);
            // 提取name属性
            String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);
            if (StringUtils.hasLength(indexAttr)) {
                try {
                    int index = Integer.parseInt(indexAttr);
                    if (index < 0) {
                        error("'index' cannot be lower than 0", ele);
                    }
                    else {
                        try {
                            this.parseState.push(new ConstructorArgumentEntry(index));
                            // 解析ele对应的属性元素
                            Object value = parsePropertyValue(ele, bd, null);
                            ConstructorArgumentValues.ValueHolder valueHolder = new ConstructorArgumentValues.ValueHolder(value);
                            if (StringUtils.hasLength(typeAttr)) {
                                valueHolder.setType(typeAttr);
                            }
                            if (StringUtils.hasLength(nameAttr)) {
                                valueHolder.setName(nameAttr);
                            }
                            valueHolder.setSource(extractSource(ele));
                            // 不允许重复指定相同参数
                            if (bd.getConstructorArgumentValues().hasIndexedArgumentValue(index)) {
                                error("Ambiguous constructor-arg entries for index " + index, ele);
                            }
                            else {
                                bd.getConstructorArgumentValues().addIndexedArgumentValue(index, valueHolder);
                            }
                        }
                        finally {
                            this.parseState.pop();
                        }
                    }
                }
                catch (NumberFormatException ex) {
                    error("Attribute 'index' of tag 'constructor-arg' must be an integer", ele);
                }
            }
            else {
                // 没有index属性则忽略此属性,自动寻找
                try {
                    this.parseState.push(new ConstructorArgumentEntry());
                    Object value = parsePropertyValue(ele, bd, null);
                    ConstructorArgumentValues.ValueHolder valueHolder = new ConstructorArgumentValues.ValueHolder(value);
                    if (StringUtils.hasLength(typeAttr)) {
                        valueHolder.setType(typeAttr);
                    }
                    if (StringUtils.hasLength(nameAttr)) {
                        valueHolder.setName(nameAttr);
                    }
                    valueHolder.setSource(extractSource(ele));
                    bd.getConstructorArgumentValues().addGenericArgumentValue(valueHolder);
                }
                finally {
                    this.parseState.pop();
                }
            }
        }
    

      上面一段看似复杂的代码让很多人失去了耐心,但是,设计的逻辑其实并不复杂,首先是提取constructor-arg上必要的属性(index type name).

    • 如果配置中指定了index属性,name操作步骤如下.
      (1) 解析constructor-arg的子元素.
      (2) 使用ConstructorArgumentValues.ValueHolder类型来封装解析出来的元素.
      (3) 将type name和index属性一并封装在ConstructorArgumentValues.ValueHolder类型中并添加至当前BeanDefinitionconstructorArgumentValues()indexedArgumentValues属性中.
    • 如果没有指定index属性,那么操作步骤如下.
      (1) 解析constructor-arg的子元素.
      (2) 使用ConstructorArgumentValues.ValueHolder类型来封装解析出来的元素.
      (3) 将type name和index属性一并封装在ConstructorArgumentValues.ValueHolder类型中并添加至当前BeanDefinitionconstructorArgumentValues()genericArgumentValues属性中.

      可以看到对于是否指定index属性来讲,Spring的处理流程是不同的,关键在于属性信息被保存的位置.
      那么了解了整个流程后,我们尝试着进一步了解解析构造函数配置中子元素的过程,进入parsePropertyValue:

    @Nullable
        public Object parsePropertyValue(Element ele, BeanDefinition bd, @Nullable String propertyName) {
            String elementName = (propertyName != null ?
                    "<property> element for property '" + propertyName + "'" :
                    "<constructor-arg> element");
    
            // Should only have one child element: ref, value, list, etc.
            // 一个属性只能对应一种类型: ref value list等
            NodeList nl = ele.getChildNodes();
            Element subElement = null;
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                // 对应description或者meta不处理
                if (node instanceof Element && !nodeNameEquals(node, DESCRIPTION_ELEMENT) &&
                        !nodeNameEquals(node, META_ELEMENT)) {
                    // Child element is what we're looking for.
                    if (subElement != null) {
                        error(elementName + " must not contain more than one sub-element", ele);
                    }
                    else {
                        subElement = (Element) node;
                    }
                }
            }
            // 解析constructor-arg上的ref属性
            boolean hasRefAttribute = ele.hasAttribute(REF_ATTRIBUTE);
            // 解析constructor-arg上的value属性
            boolean hasValueAttribute = ele.hasAttribute(VALUE_ATTRIBUTE);
            if ((hasRefAttribute && hasValueAttribute) ||
                    ((hasRefAttribute || hasValueAttribute) && subElement != null)) {
                /**
                 * 在constructor-arg上不存在 : 
                 *      1. 同时既有ref属性又有value属性
                 *      2. 存在ref属性或者value属性且又有子元素
                 */     
                error(elementName +
                        " is only allowed to contain either 'ref' attribute OR 'value' attribute OR sub-element", ele);
            }
    
            if (hasRefAttribute) {
                // ref属性的处理,使用RuntimeBeanReference封装对应的ref名称
                String refName = ele.getAttribute(REF_ATTRIBUTE);
                if (!StringUtils.hasText(refName)) {
                    error(elementName + " contains empty 'ref' attribute", ele);
                }
                RuntimeBeanReference ref = new RuntimeBeanReference(refName);
                ref.setSource(extractSource(ele));
                return ref;
            }
            else if (hasValueAttribute) {
                // value属性的处理,使用TypedStringValue封装
                TypedStringValue valueHolder = new TypedStringValue(ele.getAttribute(VALUE_ATTRIBUTE));
                valueHolder.setSource(extractSource(ele));
                return valueHolder;
            }
            else if (subElement != null) {
                // 解析子元素
                return parsePropertySubElement(subElement, bd);
            }
            else {
                // Neither child element nor "ref" or "value" attribute found.
                // 既没有ref也没有value子元素则报错
                error(elementName + " must specify a ref or value", ele);
                return null;
            }
        }
    

    从代码上来看,对构造函数中属性元素的解析,经历了一下几个过程.
    (1) 略过descriptionmeta.
    (2) 提取constructor-arg上的refvalue属性,以便于根据规则验证正确性,其规则为在constructor-arg上不存在以下情况.

    • 同时既有ref又有value属性.
    • 存在ref属性或者value属性且又有子元素.

    (3) ref属性的处理.使用RuntimeBeanReference封装对应的ref名称,如:<constructor-arg ref="a" >
    (4) value属性的处理.使用TypedStringValue封装,例如:<constructor-arg value="a" >
    (5) 子元素的处理.例如 :

    <constructor-arg>
        <map>
            <entry key="key" value="value" />
       </map>
    </constructor-arg>
    

      而对于子元素的处理,例如这里提到的在构造函数中又嵌入了子元素map是怎么实现的呢?parsePropertySubElement中实现了对各种子元素的分离处理.

    @Nullable
        public Object parsePropertySubElement(Element ele, @Nullable BeanDefinition bd) {
            return parsePropertySubElement(ele, bd, null);
        }
    
        @Nullable
        public Object parsePropertySubElement(Element ele, @Nullable BeanDefinition bd, @Nullable String defaultValueType) {
            if (!isDefaultNamespace(ele)) {
                return parseNestedCustomElement(ele, bd);
            }
            else if (nodeNameEquals(ele, BEAN_ELEMENT)) {
                BeanDefinitionHolder nestedBd = parseBeanDefinitionElement(ele, bd);
                if (nestedBd != null) {
                    nestedBd = decorateBeanDefinitionIfRequired(ele, nestedBd, bd);
                }
                return nestedBd;
            }
            else if (nodeNameEquals(ele, REF_ELEMENT)) {
                // A generic reference to any name of any bean.
                String refName = ele.getAttribute(BEAN_REF_ATTRIBUTE);
                boolean toParent = false;
                if (!StringUtils.hasLength(refName)) {
                    // A reference to the id of another bean in a parent context.
                    // 解析parent
                    refName = ele.getAttribute(PARENT_REF_ATTRIBUTE);
                    toParent = true;
                    if (!StringUtils.hasLength(refName)) {
                        error("'bean' or 'parent' is required for <ref> element", ele);
                        return null;
                    }
                }
                if (!StringUtils.hasText(refName)) {
                    error("<ref> element contains empty target attribute", ele);
                    return null;
                }
                RuntimeBeanReference ref = new RuntimeBeanReference(refName, toParent);
                ref.setSource(extractSource(ele));
                return ref;
            }
            // 对idref元素的解析
            else if (nodeNameEquals(ele, IDREF_ELEMENT)) {
                return parseIdRefElement(ele);
            }
            // 对value子元素的解析
            else if (nodeNameEquals(ele, VALUE_ELEMENT)) {
                return parseValueElement(ele, defaultValueType);
            }
            // 对null子元素的解析
            else if (nodeNameEquals(ele, NULL_ELEMENT)) {
                // It's a distinguished null value. Let's wrap it in a TypedStringValue
                // object in order to preserve the source location.
                TypedStringValue nullHolder = new TypedStringValue(null);
                nullHolder.setSource(extractSource(ele));
                return nullHolder;
            }
            else if (nodeNameEquals(ele, ARRAY_ELEMENT)) {
                // 解析array子元素
                return parseArrayElement(ele, bd);
            }
            else if (nodeNameEquals(ele, LIST_ELEMENT)) {
                // 解析list子元素
                return parseListElement(ele, bd);
            }
            else if (nodeNameEquals(ele, SET_ELEMENT)) {
                // 解析set子元素
                return parseSetElement(ele, bd);
            }
            else if (nodeNameEquals(ele, MAP_ELEMENT)) {
                // 解析map子元素
                return parseMapElement(ele, bd);
            }
            else if (nodeNameEquals(ele, PROPS_ELEMENT)) {
                // 解析props子元素
                return parsePropsElement(ele);
            }
            else {
                error("Unknown property sub-element: [" + ele.getNodeName() + "]", ele);
                return null;
            }
        }
    

      可以看到,在上面的函数中实现了所有可支持的子类的分类处理,到这里,我们已经大致理清构造函数的解析流程,至于再深入的解析大家可以自己去深入探索一下.

    7. 解析子元素property

      parsePropertyElement函数完成了对property属性的提取,property使用方式如下:

    <bean id="test" class="test.TestClass">
        <property name="testStr" value="aaa" />
    </bean>
    

    或者

    <bean id="a">
            <property name="p">
                <list>
                    <value>aa</value>
                    <value>bb</value>
                </list>
            </property>
    </bean>
    

    而具体的解析过程如下:

    public void parsePropertyElements(Element beanEle, BeanDefinition bd) {
            NodeList nl = beanEle.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                if (isCandidateElement(node) && nodeNameEquals(node, PROPERTY_ELEMENT)) {
                    parsePropertyElement((Element) node, bd);
                }
            }
        }
    

    有了之前分析方法的经验,这个方法我们并不难理解,无非是提取所有property的子元素,然后调用parsePropertyElement处理,parsePropertyElement代码如下:

    public void parsePropertyElement(Element ele, BeanDefinition bd) {
            // 获取配置元素中的name值
            String propertyName = ele.getAttribute(NAME_ATTRIBUTE);
            if (!StringUtils.hasLength(propertyName)) {
                error("Tag 'property' must have a 'name' attribute", ele);
                return;
            }
            this.parseState.push(new PropertyEntry(propertyName));
            try {
                // 不允许多次对同一属性配置
                if (bd.getPropertyValues().contains(propertyName)) {
                    error("Multiple 'property' definitions for property '" + propertyName + "'", ele);
                    return;
                }
                Object val = parsePropertyValue(ele, bd, propertyName);
                PropertyValue pv = new PropertyValue(propertyName, val);
                parseMetaElements(ele, pv);
                pv.setSource(extractSource(ele));
                bd.getPropertyValues().addPropertyValue(pv);
            }
            finally {
                this.parseState.pop();
            }
        }
    

    可以看到上面方法与构造方法注入方式不同的是返回值使用PropertyValue进行封装,并记录在了BeanDefinition中的propertyValues属性中.

    8. 解析子元素qualifier

      对于qualifier元素的获取,我们接触更多的是注解的形式,在使用Spring框架中进行自动注入时,Spring容器中匹配的候选Bean数目必须有且仅有一个.当找不到一个匹配的Bean时,Spring容器将抛出BeanCreationException异常,并指出必须至少拥有一个匹配的Bean.
      Spring允许我们通过Qualifier指定注入Bean的名称,这样歧义就消除了,而对于配置方式使用如下:

    <bean id="myTestBean" class="bean.MyTestBean">
         <qualifier type="org.springframework.beans.factory.annotation.Qualifier" value="qf" />
    </bean>
    

    其解析过程与之前大同小异,这里不再重复叙述.


    3.1.2 AbstractBeanDefinition属性

      至此我们便完成了对XML文档到GenericBeanDefinition的转换,也就是说到这里,XML中所有的配置都可以在GenericBeanDefinition的实例类中找到对应的位置.
      GenericBeanDefinition只是子类实现,而大部分的通用属性都保存在了AbstractBeanDefinition中,那么我们再次通过AbstractBeanDefinition的属性来回顾一下我们都解析了哪些对应的配置.

    public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccessor
            implements BeanDefinition, Cloneable {
    
        // 此处省略静态变量以及final常量
    
        /**
         * bean的作用范围,对应bean属性scope
         */
        @Nullable
        private String scope = SCOPE_DEFAULT;
    
        /**
         * 是否是抽象,对应bean属性abstract
         */
        private boolean abstractFlag = false;
        
        /**
         * 是否延时加载,对应bean的属性lazy-init
         */
        private boolean lazyInit = false;
    
        /**
         * 自动注入模式,对应bean属性autowire
         */
        private int autowireMode = AUTOWIRE_NO;
    
        /**
         * 依赖检查,Spring 3.0 后弃用这个属性
         */
        private int dependencyCheck = DEPENDENCY_CHECK_NONE;
    
        /**
         * 用来表示一个bean的实例化依靠另一个bean先实例化,对应bean属性depend-on
         */
        @Nullable
        private String[] dependsOn;
    
        /**
         * autowire-candidate属性设置为false,这样容器在查找自动装配对象时,将不在考虑该bean,
         * 即它不会被考虑作为其他bean自动装配的候选者,但是该bean本身还是可以使用自动装配来注入其
         * 他bean的.对应bean属性autowire-candidate
         */
        private boolean autowireCandidate = true;
    
        /**
         * 自动装配时当出现多个bean候选者时,将作为首选者,对应bean属性primary
         */
        private boolean primary = false;
    
        /**
         * 用于记录Qualifier,对应子元素qualifier
         */
        private final Map<String, AutowireCandidateQualifier> qualifiers = new LinkedHashMap<>();
    
        
        @Nullable
        private Supplier<?> instanceSupplier;
    
        /**
         * 允许访问非公开的构造器和方法,程序设置
         */
        private boolean nonPublicAccessAllowed = true;
    
        /**
         * 是否以一种宽松的模式解析构造函数,默认为true,
         * 如果为false,则在如下情况
         * interface ITest{ }
         * class ITestImpl implements ITest{ };
         * class Main{
         *    Main(ITest i){ }
         *    Main(ITestImpl i){ }   
         * }
         * 抛出异常,因为Spring无法准确定位哪个构造函数
         * 程序设置
         */
        private boolean lenientConstructorResolution = true;
        
        /**
         * 对应bean属性factory-bean
         */
        @Nullable
        private String factoryBeanName;
    
        /**
         * 对应bean属性factory-method
         */
        @Nullable
        private String factoryMethodName;
    
        /**
         * 记录构造函数注入属性,对应bean属性constructor-arg
         */
        @Nullable
        private ConstructorArgumentValues constructorArgumentValues;
    
        /**
         * 普通属性集合
         */
        @Nullable
        private MutablePropertyValues propertyValues;
    
        /**
         * 方法重写的持有者,记录lookup-method, replaced-method元素
         */
        @Nullable
        private MethodOverrides methodOverrides;
    
        /**
         * 初始化方法,对应bean属性init-method
         */
        @Nullable
        private String initMethodName;
    
        /**
         * 销毁方法,对应bean属性destory-method
         */
        @Nullable
        private String destroyMethodName;
    
        /**
         * 是否执行init-method,程序设置
         */
        private boolean enforceInitMethod = true;
    
        /**
         * 是否执行destory-method,程序设置
         */
        private boolean enforceDestroyMethod = true;
    
        /**
         * 是否是用户定义的而不是应用程序本身定义的,创建AOP时候为true,程序设置
         */
        private boolean synthetic = false;
    
        /**
         * 定义这个bean的应用
         * APPLICATION      :   用户
         * INFRASTRUCTURE   :   完全内部使用,与用户无关
         * SUPPORT          :   某些复杂配置的一部分
         * 程序设置
         */
        private int role = BeanDefinition.ROLE_APPLICATION;
    
        /**
         * bean的描述信息
         */
        @Nullable
        private String description;
    
        /**
         * 这个bean定义的资源
         */
        @Nullable
        private Resource resource;
    
        // 此处省略get/set方法
        
    }
    

    3.1.3 解析默认标签中的自定义标签元素

      到了这里我们已经完成了分析默认标签的解析与提取过程,可能因为涉及的内容太多,我们可能已经忘记了从哪个方法开始的了,我们再次回顾一下默认标签解析方法的起始方法;

    protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
            BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
            if (bdHolder != null) {
                bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
                try {
                    // Register the final decorated instance.
                    BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
                }
                catch (BeanDefinitionStoreException ex) {
                    getReaderContext().error("Failed to register bean definition with name '" +
                            bdHolder.getBeanName() + "'", ele, ex);
                }
                // Send registration event.
                getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
            }
        }
    

      我们已经用了大量的篇幅分析了BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);这句代码,接下来,我们要进行bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);代码的分析,首先大致了解了这句代码的作用,其实我们可以从语义上分析:如果需要的话就对beanDefinition进行装饰,那这句代码到底是什么功能呢?其实这句代码适用于这样的场景,如:

    <bean id="test" class="test.MyClass">
            <myBean:user userName="aaa"/>
    </bean>
    

      当Spring中的bean使用的是默认的标签配置,但是其中的子元素却使用了自定义的配置时,这句代码便会起作用了.可能有人会有疑问,之前讲过,对bean的解析分为两种类型,一种是默认类型的解析,另一种是自定义类型的解析,这不正是自定义类型的解析吗?为什么会在默认类型解析中单独添加一个方法来处理呢?确实,这个问题很让人迷惑,但是,不知道大家有没有发现,这个自定义类型并不是以Bean的形式出现的.我们之前讲过的两种类型的不同处理只是针对Bean的这里我们看到,这里我们看到,这个自定义类型其实是属性.好了,我们继续分析这段代码的逻辑.

    public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {
            return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);
        }
    

      这里将方法中第三个参数设置为空,那么第三个参数是做什么的呢?什么情况下不为空?其实第三个参数是父类bean,当对某个嵌套配置进行分析时,这里需要传递父类beanDefinition.分析源码得知这里传递的参数其实是为了使用父类的scope属性,以备子类若没有设置scope时默认使用父类的属性,这里分析的是顶层配置,所以传递null.将第三个参数设置为空后进一步跟踪函数:

    public BeanDefinitionHolder decorateBeanDefinitionIfRequired(
                Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {
    
            BeanDefinitionHolder finalDefinition = definitionHolder;
    
            // Decorate based on custom attributes first.
            NamedNodeMap attributes = ele.getAttributes();
            // 遍历所有的属性,看看是否有适用于修饰的属性
            for (int i = 0; i < attributes.getLength(); i++) {
                Node node = attributes.item(i);
                finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
            }
    
            // Decorate based on custom nested elements.
            NodeList children = ele.getChildNodes();
            // 遍历所有的子节点,看看是否有适用于修饰的子元素
            for (int i = 0; i < children.getLength(); i++) {
                Node node = children.item(i);
                if (node.getNodeType() == Node.ELEMENT_NODE) {
                    finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
                }
            }
            return finalDefinition;
        }
    

      上面的代码,我们看到方法分别对元素的所有属性以及子节点进行了decorateIfRequired的调用,我们继续跟踪代码:

    public BeanDefinitionHolder decorateIfRequired(
                Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {
            // 获取自定义标签的命名空间
            String namespaceUri = getNamespaceURI(node);
            // 对于非默认标签进行修饰
            if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {
                // 根据命名空间找到对应的处理器
                NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
                if (handler != null) {
                    // 进行修饰
                    BeanDefinitionHolder decorated =
                            handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));
                    if (decorated != null) {
                        return decorated;
                    }
                }
                else if (namespaceUri.startsWith("http://www.springframework.org/")) {
                    error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);
                }
                else {
                    // A custom namespace, not to be handled by Spring - maybe "xml:...".
                    if (logger.isDebugEnabled()) {
                        logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");
                    }
                }
            }
            return originalDef;
        }
    
        @Nullable
        public String getNamespaceURI(Node node) {
            return node.getNamespaceURI();
        }
    
    public boolean isDefaultNamespace(@Nullable String namespaceUri) {
            // BEANS_NAMESPACE_URI = "http://www.springframework.org/schema/beans";
            return (!StringUtils.hasLength(namespaceUri) || BEANS_NAMESPACE_URI.equals(namespaceUri));
        }
    

      程序走到这里,条理其实已经非常清楚了,首先获取属性或者元素的命名空间,以此来判断该元素或者属性是否适用于自定义标签的解析条件,找出自定义类型所对应的NamespaceHandler并进行进一步解析,这个以后在详细讲此处先略过.


    3.1.4 注册解析的BeanDefinition

      对于配置文件,解析也解析完了,装饰也完成了,对于得到的beanDinition已经可以满足后续的使用要求了,唯一还剩下的工作就是注册了,也就是processBeanDefinition方法中的BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());代码的解析了.

    public static void registerBeanDefinition(
                BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
                throws BeanDefinitionStoreException {
    
            // Register bean definition under primary name.
            // 使用beanName做唯一标识注册
            String beanName = definitionHolder.getBeanName();
            registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());
    
            // Register aliases for bean name, if any.
            // 注册所有别名
            String[] aliases = definitionHolder.getAliases();
            if (aliases != null) {
                for (String alias : aliases) {
                    registry.registerAlias(beanName, alias);
                }
            }
        }
    
    1. 通过beanName注册BeanDefinition

      对于beanDefinition的注册,或许很多人认为的方式就是讲beanDefinition直接放入map中就好了,使用beanName作为key.确实,Spring就是这么做的,只不过除此之外,它还做了点别的事情.

    @Override
        public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
                throws BeanDefinitionStoreException {
    
            Assert.hasText(beanName, "Bean name must not be empty");
            Assert.notNull(beanDefinition, "BeanDefinition must not be null");
    
            if (beanDefinition instanceof AbstractBeanDefinition) {
                try {
                    /* 注册前的最后一次校验,这里的校验不同于之前的XML文件校验,
                     * 主要是对于AbstractBeanDefinition属性中的methodOverrides校验
                     * 校验methodOverrides是否与工厂方法并存或者methodOverrides对应的方法根本不存在
                     */
                    ((AbstractBeanDefinition) beanDefinition).validate();
                }
                catch (BeanDefinitionValidationException ex) {
                    throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
                            "Validation of bean definition failed", ex);
                }
            }
    
            BeanDefinition existingDefinition = this.beanDefinitionMap.get(beanName);
            // 处理已经注册的beanName的情况
            if (existingDefinition != null) {
                // 如果对应的BeanName已经注册且在配置中配置了bean则不允许被覆盖(抛出异常)
                if (!isAllowBeanDefinitionOverriding()) {
                    throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
                            "Cannot register bean definition [" + beanDefinition + "] for bean '" + beanName +
                            "': There is already [" + existingDefinition + "] bound.");
                }
                else if (existingDefinition.getRole() < beanDefinition.getRole()) {
                    // e.g. was ROLE_APPLICATION, now overriding with ROLE_SUPPORT or ROLE_INFRASTRUCTURE
                    if (logger.isWarnEnabled()) {
                        logger.warn("Overriding user-defined bean definition for bean '" + beanName +
                                "' with a framework-generated bean definition: replacing [" +
                                existingDefinition + "] with [" + beanDefinition + "]");
                    }
                }
                else if (!beanDefinition.equals(existingDefinition)) {
                    if (logger.isInfoEnabled()) {
                        logger.info("Overriding bean definition for bean '" + beanName +
                                "' with a different definition: replacing [" + existingDefinition +
                                "] with [" + beanDefinition + "]");
                    }
                }
                else {
                    if (logger.isDebugEnabled()) {
                        logger.debug("Overriding bean definition for bean '" + beanName +
                                "' with an equivalent definition: replacing [" + existingDefinition +
                                "] with [" + beanDefinition + "]");
                    }
                }
                this.beanDefinitionMap.put(beanName, beanDefinition);
            }
            else {
                if (hasBeanCreationStarted()) {
                    // Cannot modify startup-time collection elements anymore (for stable iteration)
                    // 因为beanDefinitionMap是全局变量,这里定会存在并发访问的情况
                    synchronized (this.beanDefinitionMap) {
                        this.beanDefinitionMap.put(beanName, beanDefinition);
                        List<String> updatedDefinitions = new ArrayList<>(this.beanDefinitionNames.size() + 1);
                        updatedDefinitions.addAll(this.beanDefinitionNames);
                        updatedDefinitions.add(beanName);
                        this.beanDefinitionNames = updatedDefinitions;
                        if (this.manualSingletonNames.contains(beanName)) {
                            Set<String> updatedSingletons = new LinkedHashSet<>(this.manualSingletonNames);
                            updatedSingletons.remove(beanName);
                            this.manualSingletonNames = updatedSingletons;
                        }
                    }
                }
                else {
                    // Still in startup registration phase
                    // 注册beanDefinition
                    this.beanDefinitionMap.put(beanName, beanDefinition);
                    // 记录beanName
                    this.beanDefinitionNames.add(beanName);
                    this.manualSingletonNames.remove(beanName);
                }
                this.frozenBeanDefinitionNames = null;
            }
    
            if (existingDefinition != null || containsSingleton(beanName)) {
                // 重置所有beanName对应的缓存
                resetBeanDefinition(beanName);
            }
        }
    

      之前的代码中我们看到,在对于bean的注册处理方式上,主要进行了几个步骤:
     (1) 对于AbstractBeanDefinition的校验,在解析XML文件的时候我们提过校验,但是此校验非彼校验,之前的校验是针对XML格式的校验,而此时的校验是针对AbstractBeanDefinition属性中的methodOverrides的校验;
     (2) 对beanName已经注册的情况的处理,如果设置了不允许bean的覆盖,则需要抛出异常,否则直接覆盖;
     (3) 加入map缓存;
     (4) 清楚解析之前留下的对应的beanName的缓存.

    2. 通过别名注册BeanDefinition

      在理解了注册bean的原理后,理解注册别名的原理就会容易很多了.

    @Override
        public void registerAlias(String name, String alias) {
            Assert.hasText(name, "'name' must not be empty");
            Assert.hasText(alias, "'alias' must not be empty");
            // 这里也使用了锁
            synchronized (this.aliasMap) {
                // 如果beanName与alias相同的话不记录alias,并删除对应的alias
                if (alias.equals(name)) {
                    this.aliasMap.remove(alias);
                    if (logger.isDebugEnabled()) {
                        logger.debug("Alias definition '" + alias + "' ignored since it points to same name");
                    }
                }
                else {
                    String registeredName = this.aliasMap.get(alias);
                    if (registeredName != null) {
                        if (registeredName.equals(name)) {
                            // An existing alias - no need to re-register
                            return;
                        }
                        // 如果alias不允许被覆盖则抛出异常
                        if (!allowAliasOverriding()) {
                            throw new IllegalStateException("Cannot define alias '" + alias + "' for name '" +
                                    name + "': It is already registered for name '" + registeredName + "'.");
                        }
                        if (logger.isInfoEnabled()) {
                            logger.info("Overriding alias '" + alias + "' definition for registered name '" +
                                    registeredName + "' with new target name '" + name + "'");
                        }
                    }
                    // 当 A -> B 存在时,若再次出现 A -> C -> B 时则会抛出异常
                    checkForAliasCircle(name, alias);
                    this.aliasMap.put(alias, name);
                    if (logger.isDebugEnabled()) {
                        logger.debug("Alias definition '" + alias + "' registered for name '" + name + "'");
                    }
                }
            }
        }
    

      由以上代码中可以得知注册alias的步骤如下:
     (1) aliasbeanName相同情况的处理,若aliasbeanName名称相同则不需要处理并删除原有alias;
     (2) alias覆盖处理,若aliasName已经使用并已经指向了另一beanName则需要用户的设置进行处理.
     (3) alias需要检查,当 A -> B 存在时,若再次出现 A -> C -> B 时则会抛出异常.
     (4) 注册alias.


    3.1.5 通知监听器解析及注册完成

      通过代码getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));完成此工作,这里的实现只为扩展,当程序开发人员需要对注册BeanDefinition事件进行监听时可以通过注册监听器的方式并将处理逻辑写入监听器中,目前在Spring中并没有对此事做任何逻辑处理.


    3.2 alias 标签的解析

      通过上面较长的篇幅我们终于分析完了默认标签中对bean标签的处理,那么我们之前提到过,对配置文件的解析包括对import标签,alias标签,bean标签,beans标签的处理,现在我们已经完成了最重要也是最核心的功能,其他的解析步骤也都是围绕第3个解析而进行的.在分析了第3个解析步骤后,在回过头来看看对alias标签的解析.
      在对bean进行定义时,除了使用id属性来指定名称之外,为了提升多个名称,可以使用alias标签来指定.而所有的这些名称都指向同一个bean,在某些情况下提供别名非常有用,比如为了让应用的每个组件能更容易地对公共组件进行引用.
      然而,在定义bean时就指定所有的别名并不是总是恰当的.有时我们期望能在当前位置为那些在别处定义的bean引入别名.在XML配置文件中,可用单独的<alias/>元素来完成bean别名的定义.如配置文件中定义了一个javaBean:

    <bean id="testBean" class="com.test" />
    

    要给这个JavaBean增加别名,以方便不同对象来调用.我们就可以直接使用bean标签中的name属性:

    <bean id="testBean" name="testBean,testBean2" class="com.test" />
    

    同样,Spring还有另外一种声明别名的方式:

    <bean id="testBean" class="com.test" />
    <alias name="testBean" alias="testBean,testBean2" />
    

    考虑一个更为具体的例子,组件AXML配置文件中定义了一个名为componentADataSource类型的bean,但组件B却想在其XML文件中以componentB命名来引用此bean.而且在主程序MyAppXML配置文件中,希望以myApp的名字来引用此bean.最后容器加载3个XML文件来生成最终的ApplicationContext.在此情形下,可通过在配置文件中添加下列alias元素来实现:

    <alias name="componentA" alias="componentB" />
    <alias name="componentA" alias="myApp" />
    

     这样一来,每个组件及主程序就可通过唯一名字来引用同一个数据源而互不干扰.
     在之前的章节已经讲过了对于beanname元素的解析,name我们现在再来深入分析下对于alias标签的解析过程.

    protected void processAliasRegistration(Element ele) {
            // 获取beanName
            String name = ele.getAttribute(NAME_ATTRIBUTE);
            // 获取alias
            String alias = ele.getAttribute(ALIAS_ATTRIBUTE);
            boolean valid = true;
            if (!StringUtils.hasText(name)) {
                getReaderContext().error("Name must not be empty", ele);
                valid = false;
            }
            if (!StringUtils.hasText(alias)) {
                getReaderContext().error("Alias must not be empty", ele);
                valid = false;
            }
            if (valid) {
                try {
                    // 注册alias
                    getReaderContext().getRegistry().registerAlias(name, alias);
                }
                catch (Exception ex) {
                    getReaderContext().error("Failed to register alias '" + alias +
                            "' for bean with name '" + name + "'", ele, ex);
                }
                // 别名注册后通知监听器做相应的处理
                getReaderContext().fireAliasRegistered(name, alias, extractSource(ele));
            }
        }
    

      可以发现,跟之前讲过的bean中的alias解析大同小异,都是将别名与beanName组成一队注册至registry中.这里就不在展开说了.

    相关文章

      网友评论

          本文标题:第3章 默认标签的解析

          本文链接:https://www.haomeiwen.com/subject/ocbwlqtx.html