Spring源碼分析--從HelloWord開始

一丁稀、前言

學(xué)習(xí)Spring已經(jīng)有一段時間了辆脸,一直對spring的源碼比較好奇,因此開始準(zhǔn)備對Spring的源碼進行學(xué)習(xí)蕾管。這篇就從最簡單的HelloWord開始

二阔馋、分析

Java代碼


public class Person {

public Person() {

System.out.println("Person execute");

}

private String name ;

private Integer id ;

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public Integer getId() {

return id;

}

public void setId(Integer id) {

this.id = id;

}

}

applicationContext.xml

  <bean name="personName1" class="com.lw.spring.bean.Person" id="person1"></bean>

測試

  public class Test {

    @SuppressWarnings("deprecation")
    public static void main(String[] args) {

        BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));

        Person p = (Person) bf.getBean("person1");

        System.out.println(p);
    }
}

我們平常使用更多的應(yīng)該是如下的加載方式

  ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");

其實BeanFactory才是整個IOC容器的祖宗。其它的不過是對其補充娇掏。

  • XmlBeanFactory:自定義XML讀取器,下面會介紹到.
  • ApplicationContext:這其實又是一個體系的IOC容器了呕寝,只不過比較高級,后面會寫文章介紹到.

源碼分析

  public XmlBeanFactory(Resource resource) throws BeansException {
        this(resource, null);
    }
  public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
        super(parentBeanFactory);
         // 這個reader就是XMLBeanFactory對其父類DefaultListableBeanFactory的補充
        this.reader.loadBeanDefinitions(resource);
    }

我們可以看看整個XMLBeanFactory的整個代碼,可以發(fā)現(xiàn)整個XMLBeanFactory除了添加了個XmlBeanDefinitionReader 之外婴梧,并沒有添加其它什么東西.

  public class XmlBeanFactory extends DefaultListableBeanFactory {

    private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);



    public XmlBeanFactory(Resource resource) throws BeansException {
        this(resource, null);
    }


    public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
        super(parentBeanFactory);
        this.reader.loadBeanDefinitions(resource);
    }

}

下面進入到讀取器XmlBeanDefinitionReader 的代碼

    @Override
    public int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {
        return loadBeanDefinitions(new EncodedResource(resource));
    }   
  private final ThreadLocal<Set<EncodedResource>> resourcesCurrentlyBeingLoaded =
            new NamedThreadLocal<Set<EncodedResource>>("XML bean definition resources currently being loaded");
public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
    Assert.notNull(encodedResource, "EncodedResource must not be null");
    if (logger.isInfoEnabled()) {
        logger.info("Loading XML bean definitions from " + encodedResource.getResource());
    }
    // 通過本地線程獲取一個封裝EncodedResource的Set集合
    Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
    if (currentResources == null) {
        // 如果為null,則創(chuàng)建并且綁定當(dāng)前線程
        currentResources = new HashSet<EncodedResource>(4);
        this.resourcesCurrentlyBeingLoaded.set(currentResources);
    }
    // 將applicationContext.xml添加到集合當(dāng)中
    if (!currentResources.add(encodedResource)) {
        throw new BeanDefinitionStoreException(
                "Detected cyclic loading of " + encodedResource + " - check your import definitions!");
    }
    try {
        InputStream inputStream = encodedResource.getResource().getInputStream();
        try {
            InputSource inputSource = new InputSource(inputStream);
            if (encodedResource.getEncoding() != null) {
                inputSource.setEncoding(encodedResource.getEncoding());
            }
            // 加載BeanDefinition
            return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
        }
        finally {
            inputStream.close();
        }
    }
    catch (IOException ex) {
        throw new BeanDefinitionStoreException(
                "IOException parsing XML document from " + encodedResource.getResource(), ex);
    }
    finally {
        currentResources.remove(encodedResource);
        if (currentResources.isEmpty()) {
            this.resourcesCurrentlyBeingLoaded.remove();
        }
    }
}

比較重要的地方做了注釋.其中BeanDefinition其實就是<bean/>標(biāo)簽在Spring內(nèi)部的表現(xiàn)形式.

protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
        throws BeanDefinitionStoreException {
    try {
        // 加載Document文檔
        Document doc = doLoadDocument(inputSource, resource);  
        // 注冊BeanDefinition
        return registerBeanDefinitions(doc, resource);
    }
    catch (Exception ex) {
        // some exception
    }
}

上面第一步加載Document使用的sax解析下梢,就不詳細(xì)看,主要來看注冊BeanDefinition.

  public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
        BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
        int countBefore = getRegistry().getBeanDefinitionCount();
        documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
        return getRegistry().getBeanDefinitionCount() - countBefore;
    }

繼續(xù)

@Override
   public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
       this.readerContext = readerContext;
       logger.debug("Loading bean definitions");
       Element root = doc.getDocumentElement();
       doRegisterBeanDefinitions(root);
   }

 protected void doRegisterBeanDefinitions(Element root) {
    
    BeanDefinitionParserDelegate parent = this.delegate;
    this.delegate = createDelegate(getReaderContext(), root, parent);

    if (this.delegate.isDefaultNamespace(root)) {
        String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
        if (StringUtils.hasText(profileSpec)) {
            String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
                    profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
            if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
                return;
            }
        }
    }
    // 解析BeanDefinition之前應(yīng)該做的操作,空方法
    preProcessXml(root);
    // 解析BeanDefinition
    parseBeanDefinitions(root, this.delegate);
    // 解析BeanDefinition之后應(yīng)該做的操作,空方法
    postProcessXml(root);

    this.delegate = parent;
}

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
    if (delegate.isDefaultNamespace(root)) {
        NodeList nl = root.getChildNodes();
        for (int i = 0; i < nl.getLength(); i++) {
            Node node = nl.item(i);
            if (node instanceof Element) {
                Element ele = (Element) node;
                if (delegate.isDefaultNamespace(ele)) {
                    // 解析默認(rèn)標(biāo)簽
                    parseDefaultElement(ele, delegate);
                }
                else {
                    // 解析用戶自定義標(biāo)簽
                    delegate.parseCustomElement(ele);
                }
            }
        }
    }
    else {
        delegate.parseCustomElement(root);
    }
}

上面解析默認(rèn)標(biāo)簽其實就是bean塞蹭、import之類的了孽江。解析標(biāo)簽時候同時也會解析其屬性,由于太過復(fù)雜番电,就沒有深入下去了岗屏。

BeanDefinition

上面提到過<bean/>標(biāo)簽在Spring內(nèi)部的表現(xiàn)形式其實就是BeanDefinition,而BeanDefinition只是一個接口漱办,它有哪些實現(xiàn)類呢这刷?

  • ChildBeanDefinition
  • GenericBeanDefinition
  • RootBeanDefinition

其中RootBeanDefinition就是我們通常的bean了。

OK娩井、這篇就簡單到這里了暇屋。其實這里只是簡單的分析一行代碼,下一篇將會分析Bean是如何獲取的洞辣。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末咐刨,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子扬霜,更是在濱河造成了極大的恐慌定鸟,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,496評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件著瓶,死亡現(xiàn)場離奇詭異联予,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,187評論 3 385
  • 文/潘曉璐 我一進店門躯泰,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人华糖,你說我怎么就攤上這事麦向。” “怎么了客叉?”我有些...
    開封第一講書人閱讀 157,091評論 0 348
  • 文/不壞的土叔 我叫張陵诵竭,是天一觀的道長。 經(jīng)常有香客問我兼搏,道長卵慰,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,458評論 1 283
  • 正文 為了忘掉前任佛呻,我火速辦了婚禮裳朋,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吓著。我一直安慰自己鲤嫡,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,542評論 6 385
  • 文/花漫 我一把揭開白布绑莺。 她就那樣靜靜地躺著暖眼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪纺裁。 梳的紋絲不亂的頭發(fā)上诫肠,一...
    開封第一講書人閱讀 49,802評論 1 290
  • 那天,我揣著相機與錄音欺缘,去河邊找鬼栋豫。 笑死,一個胖子當(dāng)著我的面吹牛谚殊,可吹牛的內(nèi)容都是我干的笼才。 我是一名探鬼主播,決...
    沈念sama閱讀 38,945評論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼络凿,長吁一口氣:“原來是場噩夢啊……” “哼骡送!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起絮记,我...
    開封第一講書人閱讀 37,709評論 0 266
  • 序言:老撾萬榮一對情侶失蹤摔踱,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后怨愤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體派敷,經(jīng)...
    沈念sama閱讀 44,158評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,502評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了篮愉。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片腐芍。...
    茶點故事閱讀 38,637評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖试躏,靈堂內(nèi)的尸體忽然破棺而出猪勇,到底是詐尸還是另有隱情,我是刑警寧澤颠蕴,帶...
    沈念sama閱讀 34,300評論 4 329
  • 正文 年R本政府宣布泣刹,位于F島的核電站,受9級特大地震影響犀被,放射性物質(zhì)發(fā)生泄漏椅您。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,911評論 3 313
  • 文/蒙蒙 一寡键、第九天 我趴在偏房一處隱蔽的房頂上張望掀泳。 院中可真熱鬧,春花似錦西轩、人聲如沸开伏。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,744評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽固灵。三九已至,卻和暖如春劫流,著一層夾襖步出監(jiān)牢的瞬間巫玻,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,982評論 1 266
  • 我被黑心中介騙來泰國打工祠汇, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留仍秤,地道東北人。 一個月前我還...
    沈念sama閱讀 46,344評論 2 360
  • 正文 我出身青樓可很,卻偏偏與公主長得像诗力,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子我抠,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,500評論 2 348

推薦閱讀更多精彩內(nèi)容