Spring中的BeanFactory工廠詳細(xì)解析
BeanFactory
Spring的本質(zhì)是一個(gè)bean工廠(beanFactory)或者說(shuō)bean容器,它按照我們的要求,生產(chǎn)我們需要的各種各樣的bean,提供給我們使用。
只是在生產(chǎn)bean的過(guò)程中,需要解決bean之間的依賴問(wèn)題,才引入了依賴注入(DI)這種技術(shù)。
也就是說(shuō)依賴注入是beanFactory生產(chǎn)bean時(shí)為了解決bean之間的依賴的一種技術(shù)而已。
beanFactory會(huì)在bean的生命周期的各個(gè)階段中對(duì)bean進(jìn)行各種管理,并且spring將這些階段通過(guò)各種接口暴露給我們,讓我們可以對(duì)bean進(jìn)行各種處理,我們只要讓bean實(shí)現(xiàn)對(duì)應(yīng)的接口,那么spring就會(huì)在bean的生命周期調(diào)用我們實(shí)現(xiàn)的接口來(lái)處理該bean.
在介紹BeanFactory前,我們先想一個(gè)問(wèn)題。
我們都知道BeanFactory主要的內(nèi)容是幫我們生成Bean信息和管理Bean信息,那么我們?cè)趚ml文件中<baen>屬性的時(shí)候。
Spring是如何幫我們生成Bean的
BeanDefinitionReader 從xml文件、類路徑下使用了@Component系列注解的類、或者從@Configuration注解的配置類,獲取BeanDefintiions,然后注冊(cè)到BeanFactory中。
我們通過(guò)BeanDefinitionReader從不同的方式,獲取BeanDefintiions Bean的元數(shù)據(jù)。
BeanDefintiions是什么
protected AbstractBeanDefinition(BeanDefinition original) {
setParentName(original.getParentName());
setBeanClassName(original.getBeanClassName());
setScope(original.getScope());
setAbstract(original.isAbstract());
setFactoryBeanName(original.getFactoryBeanName());
setFactoryMethodName(original.getFactoryMethodName());
setRole(original.getRole());
setSource(original.getSource());
copyAttributesFrom(original);
。。。。。。省略
}從上面的代碼我們大致的看出。BeanDefintiions其實(shí)就是對(duì)Bean的一些元數(shù)據(jù)定義
包括parenName 父類名稱 baenClassName:類名,scope bean的作用域。Abstract是否是抽象的等信息。
通過(guò) BeanDefinitionReader獲取到BeanDefinition之后 。
我們?cè)谕ㄟ^(guò)BeanDefinitionRegistry將beanDefinition注冊(cè)到BeanFacory中。
存儲(chǔ)在BeanFactory的一個(gè)conCurrentHashMap中。key為beanName,Value就是BeanDefinition元數(shù)據(jù)。
那么獲取Bean就從conCurrentHashMap中通過(guò)BeanName獲取對(duì)應(yīng)的Bean信息。
總結(jié)
從上面的分析:我們可以看到Bean的加載解析過(guò)程如下圖所示:

接下來(lái)我們針對(duì)BeanDefinitionReader、BeanDefinitionRegistry、BeanFactory分別分析:
BeanDefinitionReader
1. XmlBeanDefinitionReader:
基于XML文件 讀取解析xml文件,通過(guò)Parser解析xml文件的標(biāo)簽。針對(duì)beans標(biāo)簽,生成對(duì)應(yīng)的BeanDefintions,然后注冊(cè)到BeanFactory中。 針對(duì)其他有特殊功能的標(biāo)簽,如context:component-scan,context:anotation-config,還可以生成BeanFactoryPostProcessor,BeanPostProcessor接口實(shí)現(xiàn)類的bean等;除了可以生成BeanDefinitions之外,還可以實(shí)現(xiàn)其他功能。NamespaceHandler:XML標(biāo)簽名稱空間處理器 被XmlBeanDefinitionReader使用,XmlBeanDefinitionReader在處理每個(gè)XML標(biāo)簽名稱空間的時(shí)候,如applicationContext.xml的context:,mvc:,通過(guò)一個(gè)DefaultNamespaceHandlerResolver來(lái)獲取對(duì)應(yīng)的NamespaceHandler實(shí)現(xiàn)類,然后通過(guò)這個(gè)NamespaceHandler實(shí)現(xiàn)類,進(jìn)一步獲取該命名空間的內(nèi)部標(biāo)簽對(duì)應(yīng)的BeanDefinitionParser實(shí)現(xiàn)類。 被XmlBeanDefinitionReader使用,專門(mén)用于處理xml文件的beans標(biāo)簽的標(biāo)簽處理器。即XmlBeanDefinitionReader讀取xml文件,創(chuàng)建Document對(duì)象,然后交給BeanDefinitionDocumentReader處理。BeanDefinitionDocumentReader解析Document對(duì)象的Element節(jié)點(diǎn),然后創(chuàng)建BeanDefinitions集合,通過(guò)XmlBeanDefinitionReader注冊(cè)到XmlBeanDefinitionReader所在的BeanFactory。
2. AnnotatedBeanDefinitionReader:
注冊(cè)指定的類列表annotatedClasses 可以使用編程方法,顯示指定將哪些類需要注冊(cè)到BeanFactory。 主要是被AnnotationConfigApplicationContext使用,即基于注解配置的ApplicationContext,這是SpringBoot的默認(rèn)ApplicationContext。典型使用為:先獲取所有使用了@Configuration注解的類,然后通過(guò)AnnotatedBeanDefinitionReader生成與這些類對(duì)應(yīng)的BeanDefinitions,并注冊(cè)到BeanFactory。
3. ClassPathBeanDefinitionScanner:
注冊(cè)指定的basePackages下面的類 掃描指定類路徑(包)下面的類,檢測(cè)是否存在@Component注解及其子注解,從而生成BeanDefinition,然后注冊(cè)到BeanFactory。沒(méi)有實(shí)現(xiàn)BeanDefinitionReader接口,但基于相同的設(shè)計(jì)思路:BeanDefinitionReader。與AnnotatedBeanDefinitionReader一樣,都是獲取指定類,生成該類的BeanDefinition注冊(cè)到BeanFactory,而不是像xml文件一樣已經(jīng)通過(guò)bean標(biāo)簽顯示說(shuō)明這個(gè)就是bean。也是主要是被AnnotationConfigApplicationContext使用。與其他兩種也是基于basePackages類路徑掃描的方式不同之處為:context:component-scan標(biāo)簽:基于XML的ApplicationContext,實(shí)現(xiàn)類路徑掃描,底層使用ComponentScanBeanDefinitionParser這個(gè)parser來(lái)處理。@ComponentScan:對(duì)于處理@Configuration配置類上面的@ComponentScan注解,則是通過(guò)ComponentScanAnnotationParser來(lái)處理的。
4. ConfigurationClassBeanDefinitionReader:
基于@Configuration注解的類配置 處理@Configuration注解的配置類,加在這些配置類上面的注解,即與@Configuration一起使用的注解,如@ComponentScan,@PropertySource,@Import,@Profile等。 ConfigurationClassBeanDefinitionReader主要被ConfigurationClassPostProcessor調(diào)用,ConfigurationClassPostProcessor為BeanFactoryPostProcessor
BeanDefinitionRegistry
注冊(cè)BeanDefinitions。提供registerBeanDefinition,removeBeanDefinition等方法,用來(lái)從BeanFactory注冊(cè)或移除BeanDefinition。
通常BeanFactory接口的實(shí)現(xiàn)類需要實(shí)現(xiàn)這個(gè)接口。
實(shí)現(xiàn)類(通常為BeanFactory接口實(shí)現(xiàn)類)的對(duì)象實(shí)例,被DefinitionReader接口實(shí)現(xiàn)類引用,DefinitionReader將BeanDefintion注冊(cè)到該對(duì)象實(shí)例中。
除了上述的BeanDefinitionRegisry還有一個(gè)負(fù)責(zé)單例Bean注冊(cè)的接口:SingletonBeanRegistry
用于注冊(cè)單例Bean對(duì)象實(shí)例,實(shí)現(xiàn)類定義存儲(chǔ)Bean對(duì)象實(shí)例的map,BeanFactory的類層次結(jié)構(gòu)中需要實(shí)現(xiàn)這個(gè)接口,來(lái)提供Bean對(duì)象的注冊(cè)和從Bean對(duì)象實(shí)例的map獲取bean對(duì)象。
BeanFactory
接口具體實(shí)現(xiàn)類
1. DefaultListableBeanFactory
BeanFactory接口體系的默認(rèn)實(shí)現(xiàn)類,實(shí)現(xiàn)以上接口的功能,提供BeanDefinition的存儲(chǔ)map,Bean對(duì)象對(duì)象的存儲(chǔ)map。
其中Bean對(duì)象實(shí)例的存儲(chǔ)map,定義在FactoryBeanRegistrySupport,FactoryBeanRegistrySupport實(shí)現(xiàn)了SingletonBeanRegistry接口,而DefaultListableBeanFactory的基類AbstractBeanFactory,繼承于FactoryBeanRegistrySupport。
2. StaticListableBeanFactory
用于存儲(chǔ)給定的bean對(duì)象實(shí)例,不支持動(dòng)態(tài)注冊(cè)功能,是ListableBeanFactory接口的簡(jiǎn)單實(shí)現(xiàn)。
beanFactoty后置處理器: BeanFactoryPostProcessor
benFactoryPostProCessor是BeanFactory的后置處理器:
在BeanFactory創(chuàng)建好,加載好其所包含的所有beanDefinitions,但是還沒(méi)有實(shí)例化bean之前,執(zhí)行,具體為調(diào)用postProcessBeanFactory方法。
1. 加載更多的bean元數(shù)據(jù) ConfigurationClassPostProcessor,用于從BeanFactory中檢測(cè)使用了@Configuration注解的類,對(duì)于這些類對(duì)應(yīng)的BeanDefinitions集合,遍歷并依次交給ConfigurationClassParser,ConfigurationClassBeanDefinitionReader處理,分別是處理與@Configuration同時(shí)使用的其他注解和將類內(nèi)部的使用@Bean注解的方法,生成BeanDefinition,注冊(cè)到BeanFactory。
2. 對(duì)bean元數(shù)據(jù)進(jìn)行加工處理 BeanDefinition屬性填充、修改:在postProcessBeanFactory方法中,可以對(duì)beanFactory所包含的beanDefinitions的propertyValues和構(gòu)造函數(shù)參數(shù)值進(jìn)行修改,如使用PropertyPlaceHolderConfigurer來(lái)對(duì)BeanDefinition的propertyValues的占位符進(jìn)行填充、賦值?;蛘呤褂肞ropertyResourceConfigurer獲取config文件中屬性,對(duì)BeanDefinitions的相關(guān)屬性進(jìn)行賦值或者值覆蓋。
bean對(duì)象后置處理器:BeanPostProcessor
Bean后置處理器:負(fù)責(zé)對(duì)已創(chuàng)建好的bean對(duì)象進(jìn)行加工處理。
主要是可以對(duì)新創(chuàng)建的bean實(shí)例進(jìn)行修改,提供了一個(gè)類似于hook機(jī)制,對(duì)創(chuàng)建好的bean對(duì)象實(shí)例進(jìn)行修改。
核心方法postProcessBeforeInitialization:在創(chuàng)建好bean實(shí)例,但是在任何初始化回調(diào)執(zhí)行之前,如InitializingBean的afterPropertiesSet,先執(zhí)行該方法。
postProcessAfterInitialization:在創(chuàng)建好bean實(shí)例,并且所有的初始化回調(diào)都執(zhí)行完了,如InitializingBean的afterPropertiesSet,再執(zhí)行該方法。
至此:BeanFactory的Bean加載過(guò)程全部分析完成:
到此這篇關(guān)于Spring中的BeanFactory工廠詳細(xì)解析的文章就介紹到這了,更多相關(guān)BeanFactory工廠內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java判斷中文字符串長(zhǎng)度的簡(jiǎn)單實(shí)例
下面小編就為大家?guī)?lái)一篇java判斷中文字符串長(zhǎng)度的簡(jiǎn)單實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01
Spring使用hutool的HttpRequest發(fā)送請(qǐng)求的幾種方式
Spring HttpRequest是指Spring框架中的一個(gè)對(duì)象,它代表了HTTP客戶端發(fā)送給Web服務(wù)器的一次請(qǐng)求,本文給大家介紹了Spring使用hutool的HttpRequest發(fā)送請(qǐng)求的幾種方式,并通過(guò)代碼示例講解的非常詳細(xì),需要的朋友可以參考下2024-11-11
Maven清理java項(xiàng)目中未使用到 jar 依賴包的方法
本文主要介紹了Maven清理java項(xiàng)目中未使用到 jar 依賴包的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-02-02
詳解IDEA使用Maven項(xiàng)目不能加入本地Jar包的解決方法
這篇文章主要介紹了詳解IDEA使用Maven項(xiàng)目不能加入本地Jar包的解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Java 調(diào)用 HTTP 接口的 7 種方式示例代碼(全網(wǎng)最全指南)
在開(kāi)發(fā)過(guò)程中,調(diào)用 HTTP 接口是最常見(jiàn)的需求之一,本文將詳細(xì)介紹 Java 中 7 種主流的調(diào)用 HTTP 接口的方式,包括每種工具的優(yōu)缺點(diǎn)和完整代碼實(shí)現(xiàn),感興趣的朋友一起看看吧2025-02-02
IDEA 中創(chuàng)建Spring Data Jpa 項(xiàng)目的示例代碼
這篇文章主要介紹了IDEA 中創(chuàng)建Spring Data Jpa 項(xiàng)目的示例代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04
解決IDEA2021版compiler.automake.allow.when.app.running不存在的問(wèn)題
很多文章介紹IntelliJ IDEA開(kāi)啟熱部署功能都會(huì)寫(xiě)到在IntelliJ IDEA中的注冊(cè)表中開(kāi)啟compiler.automake.allow.when.app.running選項(xiàng),此選項(xiàng)在IntelliJ IDEA 2021.2之后的版本遷移到高級(jí)設(shè)置中,下面看下設(shè)置方法2021-09-09

