美文网首页
Spring IOC的深入理解(四)

Spring IOC的深入理解(四)

作者: chengcongyue | 来源:发表于2019-04-15 12:11 被阅读0次

    引言

    本章开始重点研究,默认便签的解析

        private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {
            if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) {
                importBeanDefinitionResource(ele);
            }
            else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) {
                processAliasRegistration(ele);
            }
            else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) {
                processBeanDefinition(ele, delegate);
            }
            else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) {
                // recurse
                doRegisterBeanDefinitions(ele);
            }
        }
    

    接着上一章


    图片.png

    我们开始研究默认便签是如何变成BeanDefinition的,上面的代码我们先研究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));
            }
        }
    

    BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
    bdHolder 就包括我们bean中的各种属性了,如果bean下还有其他的自定义结点,还需要进一步的解析.解析完了就需要对bdHolder 进行注册,最后发出一个相应时间,说明这个bean标签以及加载好了
    然后我们仔细分析:
    parseBeanDefinitionElement

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

    parseBeanDefinitionElement的方法代码量过大,我们分开写

    String id = ele.getAttribute(ID_ATTRIBUTE);
    String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);
    

    提取元素的id和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.isTraceEnabled()) {
                    logger.trace("No XML 'id' specified - using '" + beanName +
                            "' as bean name and " + aliases + " as aliases");
                }
            }
    
            if (containingBean == null) {
                checkNameUniqueness(beanName, aliases, ele);
            }
    

    分割name属性

    AbstractBeanDefinition beanDefinition = 
    parseBeanDefinitionElement(ele, beanName, containingBean);
    

    这一步的把其他的属性统一封装到GenericBeanDefinition的实例之中
    我们看一看这个代码

    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 {
                    //创建GenericBeanDefinition用来承载属性
                AbstractBeanDefinition bd = createBeanDefinition(className, parent);
                             //硬编码解析默认的bean的各种属性
                parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
                bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));
    //解析元数据
                parseMetaElements(ele, bd);
    //两个不常见的紫属性
                parseLookupOverrideSubElements(ele, bd.getMethodOverrides());
                parseReplacedMethodSubElements(ele, bd.getMethodOverrides());
    //解析构造函数
                parseConstructorArgElements(ele, bd);
    //property
                parsePropertyElements(ele, bd);
    //....
                parseQualifierElements(ele, bd);
    
                bd.setResource(this.readerContext.getResource());
                bd.setSource(extractSource(ele));
    
                return bd;
        ......
    

    AbstractBeanDefinition bd = createBeanDefinition(className, parent);这一句代码十分让人在意.我们来了解BeanDefinition.BeanDefinition的作用就是属性承载
    BeanDefinition是一个接口,在spring中有三种实现方式,RootBeanDefinition,ChildBeanDefinition,GenericBeanDefinition.这三个都继承了AbstractBeanDefinition.BeanDefinition是<bean>在容器内部的表现形式.
    Spring将<bean>转化成BeanDefinition在内容内存在,并把不同的BeanDefinition注册到BeanDefinitionRegistry中.BeanDefinitionRegistry就像是Spring容器的数据库,主要是以map形式保存的,后续操作直接从BeanDefinitionRegistry中读取信息

    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();
            bd.setParentName(parentName);
            if (className != null) {
                if (classLoader != null) {
                    bd.setBeanClass(ClassUtils.forName(className, classLoader));
                }
                else {
                    bd.setBeanClassName(className);
                }
            }
            return bd;
        }
    
    
    
    

    现在已经创建好了承载bean信息的实例之后,然后就是对bean信息进行解析了


    图片.png
    parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
    

    进入到这个方法中,代码量过大,大概意思就是解析各种不同的属性.
    至此我们就完成了xml文档到GenericBeanDefinition的转换,XML在GenericBeanDefinition中都能找到对应的配置.

    BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
    

    我们已经分析了很多篇幅的上面的代码,现在我们来分析一下

    bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
    
    public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, 
    

    对应程序默认的便签是直接略过的,对于默认的便签在这里已经处理完成了,在方法中寻找自定义便签,并对自定义标签寻找命名空间,并进行进一步的解析

    然后就是剩下的工作就是注册了

    BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
    
    public static void registerBeanDefinition(
                BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
                throws BeanDefinitionStoreException {
    
            // Register bean definition under primary name.
            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);
                }
            }
        }
    

    其中分为两个步骤,一个是对beanName的注册,一个是对别名的注册

    • 对beanName的注册,就是将beanDefinition直接放入map,并将BeanName作为key存放
      在此之前会做一步校验,主要是针对methodOverride的,如果当前的BeanName已经注册,这时候methodOverride设置了不允许覆盖,就会抛出异常.
      然后加入map
      清除之前的缓存.
    • 别名的注册类似

    相关文章

      网友评论

          本文标题:Spring IOC的深入理解(四)

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