spring源碼(2)Register the bean definitions(1)

上一節:http://blog.csdn.net/disiwei1012/article/details/75452356

繼續上次的分析:


XmlBeanDefinitionReader:

@SuppressWarnings("deprecation")
    public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
        BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
        //設置上下文環境
        documentReader.setEnvironment(getEnvironment());
        //記錄加載的beanDefinition個數
        int countBefore = getRegistry().getBeanDefinitionCount();
        //註冊bean
        documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
        //記錄本次加載的beanDefinition個數
        return getRegistry().getBeanDefinitionCount() - countBefore;
    }

創建DocumentReader (createBeanDefinitionDocumentReader),允許用戶自定義DocumentReader,但必須要實現BeanDefinitionDocumentReader接口。

protected BeanDefinitionDocumentReader createBeanDefinitionDocumentReader() {
        return BeanDefinitionDocumentReader.class.cast(BeanUtils.instantiateClass(this.documentReaderClass));
    }

注:
1.beanDefinition 相當於XML中的bean標籤
2.BeanDefinitionDocumentReader.class.cast(Obj A) :強制將A轉化成BeanDefinitionDocumentReader類型

public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
        this.readerContext = readerContext;
        logger.debug("Loading bean definitions");
        Element root = doc.getDocumentElement();
        //真正開始解析的地方
        doRegisterBeanDefinitions(root);
    }



/**
     * Register each bean definition within the given root {@code <beans/>} element.
     */
    protected void doRegisterBeanDefinitions(Element root) {
        String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
        if (StringUtils.hasText(profileSpec)) {
            String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
                    profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
            if (!getEnvironment().acceptsProfiles(specifiedProfiles)) {
                return;
            }
        }

        // Any nested <beans> elements will cause recursion in this method. In
        // order to propagate and preserve <beans> default-* attributes correctly,
        // keep track of the current (parent) delegate, which may be null. Create
        // the new (child) delegate with a reference to the parent for fallback purposes,
        // then ultimately reset this.delegate back to its original (parent) reference.
        // this behavior emulates a stack of delegates without actually necessitating one.
        BeanDefinitionParserDelegate parent = this.delegate;
        //解析代理對象
        this.delegate = createDelegate(this.readerContext, root, parent);

        preProcessXml(root);
        //開始解析
        parseBeanDefinitions(root, this.delegate);
        postProcessXml(root);

        this.delegate = parent;
    }

註冊根節點root()中的每一個beanDefinition。

1.首先獲取根節點beans的profile屬性值。值與值之間可以使用逗號或者分號分隔。

2.然後判斷profile是否都符合環境變量中所定義的,不符合則不會去解析。

假如:
acceptsProfiles(“dev”,”pro”) :如果dev和pro均被激活則返回true,否則返回false
acceptsProfiles(“dev”,”!pro”) :如果dev被激活,pro未被激活返回true,否則返回false
激活的方式有:
①JVM參數

set JAVA_OPTS="-Dspring.profiles.active=pro"

②web.xml

<init-param>
  <param-name>spring.profiles.active</param-name>
  <param-value>production</param-value>
</init-param>

等等。

3.創建解析代理對象delegate,將解析工作委託給delegate

4.preProcessXml、postProcessXml方法體均爲空,模板方法模式。

parseBeanDefinitions方法是真正解析開始的地方!

模板方法模式請參考:http://blog.csdn.net/disiwei1012/article/details/53467251

代理模式請參考:http://blog.csdn.net/disiwei1012/article/details/53440501

對於beans的profile具體用法參考我的這篇博客:http://blog.csdn.net/disiwei1012/article/details/75716369

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章