RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Spring中BeanFactory和FactoryBean的區(qū)別在哪?

OSC開(kāi)源社區(qū) ? 來(lái)源:OSC開(kāi)源社區(qū) ? 2023-07-13 09:03 ? 次閱讀

其實(shí)從名字上就能看出來(lái)個(gè)一二,BeanFactory 是 Factory 而 FactoryBean 是一個(gè) Bean,我們先來(lái)看下總結(jié):

BeanFactory 是 Spring 框架的核心接口之一,用于管理和獲取應(yīng)用程序中的 Bean 實(shí)例。它是一個(gè)工廠模式的實(shí)現(xiàn),負(fù)責(zé)創(chuàng)建、配置和管理 Bean 對(duì)象。BeanFactory 是 Spring IoC 容器的基礎(chǔ),它可以從配置元數(shù)據(jù)(如 XML 文件)中讀取 Bean 的定義,并在需要時(shí)實(shí)例化和提供這些 Bean。

FactoryBean 是一個(gè)特殊的 Bean,它是一個(gè)工廠對(duì)象,用于創(chuàng)建和管理其他 Bean 的實(shí)例。FactoryBean 接口定義了一種創(chuàng)建 Bean 的方式,它允許開(kāi)發(fā)人員在 Bean 的創(chuàng)建過(guò)程中進(jìn)行更多的自定義操作。通過(guò)實(shí)現(xiàn) FactoryBean 接口,開(kāi)發(fā)人員可以創(chuàng)建復(fù)雜的 Bean 實(shí)例,或者在 Bean 實(shí)例化之前進(jìn)行一些額外的邏輯處理。

區(qū)別在于,BeanFactory 是 Spring 框架的核心接口,用于管理和提供 Bean 實(shí)例,而 FactoryBean 是一個(gè)特殊的 Bean,用于創(chuàng)建和管理其他 Bean 的實(shí)例。FactoryBean 在 Bean 的創(chuàng)建過(guò)程中提供更多的自定義能力,允許進(jìn)行額外的邏輯處理。

可能有的小伙伴看的還不是很清楚,我們?cè)賮?lái)詳細(xì)看下。

1. BeanFactory

BeanFactory 看名字就知道這是一個(gè) Bean 工廠,小伙伴們知道,Spring IoC 容器幫我們完成了 Bean 的創(chuàng)建、管理等操作,那么這些操作都離不開(kāi) BeanFactory。

我們來(lái)簡(jiǎn)單看下 BeanFactory 的代碼:

publicinterfaceBeanFactory{
StringFACTORY_BEAN_PREFIX="&";
ObjectgetBean(Stringname)throwsBeansException;
TgetBean(Stringname,ClassrequiredType)throwsBeansException;
ObjectgetBean(Stringname,Object...args)throwsBeansException;
TgetBean(ClassrequiredType)throwsBeansException;
TgetBean(ClassrequiredType,Object...args)throwsBeansException;
ObjectProvidergetBeanProvider(ClassrequiredType);
ObjectProvidergetBeanProvider(ResolvableTyperequiredType);
booleancontainsBean(Stringname);
booleanisSingleton(Stringname)throwsNoSuchBeanDefinitionException;
booleanisPrototype(Stringname)throwsNoSuchBeanDefinitionException;
booleanisTypeMatch(Stringname,ResolvableTypetypeToMatch)throwsNoSuchBeanDefinitionException;
booleanisTypeMatch(Stringname,ClasstypeToMatch)throwsNoSuchBeanDefinitionException;
@Nullable
ClassgetType(Stringname)throwsNoSuchBeanDefinitionException;
@Nullable
ClassgetType(Stringname,booleanallowFactoryBeanInit)throwsNoSuchBeanDefinitionException;
String[]getAliases(Stringname);

}

這些方法基本上都見(jiàn)名知義:

FACTORY_BEAN_PREFIX:這個(gè)變量其實(shí)是說(shuō),如果當(dāng)前 Bean 不是像 User、Book 這種普通 Bean,而是一個(gè) FactoryBean,就給這個(gè) Bean 名字加一個(gè) & 前綴,這個(gè)我在第二小節(jié)和小伙伴們演示。

getBean:這個(gè)方法就是根據(jù) Bean 的名字、類(lèi)型等去查詢 Bean。

getBeanProvider:這個(gè)方法可以獲取一個(gè) ObjectProvider,ObjectProvider 是 Spring 框架中的一個(gè)接口,用于獲取 Bean 對(duì)象的實(shí)例。它提供了一種延遲加載 Bean 的方式,可以在需要時(shí)動(dòng)態(tài)地獲取 Bean 實(shí)例(懶加載)。

containsBean:判斷是否包含某個(gè) Bean。

isSingleton:判斷某個(gè) Bean 是否是單例的。

isPrototype:判斷某個(gè) Bean 是否是多例的。

isTypeMatch:判斷某一個(gè) Bean 的類(lèi)型是否是給定類(lèi)型。

getType:獲取 Bean 的類(lèi)型。

getAliases:獲取 Bean 的別名。

可以看到,很多都是大家日常開(kāi)發(fā)中常見(jiàn)常用的方法。

很多小伙伴剛開(kāi)始接觸 Spring 的時(shí)候,都會(huì)用到一個(gè)對(duì)象 ClassPathXmlApplicationContext,這其實(shí)就是 BeanFactory 的一個(gè)子類(lèi)。我們來(lái)看下 BeanFactory 的繼承圖:

e45dc2fe-20a0-11ee-962d-dac502259ad0.png

繼承類(lèi)比較多,我說(shuō)幾個(gè)大家可能比較熟悉的:

ClassPathXmlApplicationContext:這個(gè)是 Spring 容器啟動(dòng)時(shí),從當(dāng)前類(lèi)路徑下去加載 XML 配置文件,參數(shù)就是 classpath 下 XML 的文件路徑。

FileSystemXmlApplicationContext:這個(gè)是 Spring 容器啟動(dòng)時(shí),從文件系統(tǒng)中去加載 XML 配置文件,參數(shù)一個(gè)絕對(duì)路徑。

AnnotationConfigApplicationContext:這個(gè)是如果我們使用 Java 代碼去做 Spring 容器的配置的話,通過(guò)這個(gè)配置類(lèi)去加載 Java 配置類(lèi)。

DefaultListableBeanFactory:這個(gè)默認(rèn)實(shí)現(xiàn)了 ListableBeanFactory 和 BeanDefinitionRegistry 接口,是一個(gè)比較成熟的 BeanFactory。

好啦,這就是 BeanFactory 的特點(diǎn),大家明白了吧~

2. FactoryBean

2.1 分析

FactoryBean 其實(shí)很多小伙伴可能都見(jiàn)過(guò),只是可能沒(méi)去總結(jié)歸納。我給小伙伴們舉幾個(gè)例子。

在 SSM 項(xiàng)目中,如果我們要配置 MyBatis 到項(xiàng)目中,一般需要配置下面這個(gè) Bean:






classpath*:org/javaboy/shirodemo/mapper/*.xml



我們?cè)谂渲?Shiro 的時(shí)候,一般都要配置如下 Bean:








/index=anon
/doLogin=anon
/hello=user
/**=authc



如果我們前端傳遞的參數(shù)是 key-value 格式,并且有一個(gè)日期,那么小伙伴們知道,服務(wù)端 SpringMVC 默認(rèn)無(wú)法處理這個(gè)日期,需要配置一個(gè)日期轉(zhuǎn)換器,一般我們?cè)?Spring 容器中添加如下 Bean:









小伙伴們觀察上面三個(gè) Bean 有一個(gè)共同的特點(diǎn),那就是 Bean 的名字都是 xxxFactoryBean。

為什么要用 xxxFactoryBean 而不直接把需要的 Bean 注入到 Spring 容器中去呢?以 MyBatis 為例:

手動(dòng)配置過(guò) MyBatis 的小伙伴應(yīng)該都知道,MyBatis 有兩個(gè)重要的類(lèi),一個(gè)是 SqlSessionFactory,還有一個(gè)是 SqlSession,通過(guò) SqlSessionFactory 可以獲取到一個(gè) SqlSession。但是不知道小伙伴們是否還記得配置代碼,手動(dòng)配置代碼如下:

publicclassSqlSessionFactoryUtils{
privatestaticSqlSessionFactorySQLSESSIONFACTORY=null;
publicstaticSqlSessionFactorygetInstance(){
if(SQLSESSIONFACTORY==null){
try{
SQLSESSIONFACTORY=newSqlSessionFactoryBuilder().build(Resources.getResourceAsStream("mybatis-config.xml"));
}catch(IOExceptione){
e.printStackTrace();
}
}
returnSQLSESSIONFACTORY;
}
}
publicclassMain{
publicstaticvoidmain(String[]args){
SqlSessionFactoryfactory=SqlSessionFactoryUtils.getInstance();
SqlSessionsqlSession=factory.openSession();
Listlist=sqlSession.selectList("org.javaboy.mybatis01.mapper.UserMapper.getAllUser");
for(Useruser:list){
System.out.println("user="+user);
}
sqlSession.close();
}
}

小伙伴們看到,無(wú)論是 SqlSessionFactory 還是 SqlSession,都不是正經(jīng) new 出來(lái)的,其實(shí)這兩個(gè)都是接口,顯然不可能 new 出來(lái),前者通過(guò)建造者模式去配置各種屬性,最后生成一個(gè) SqlSessionFactory 的實(shí)例,后者則通過(guò)前者這個(gè)工廠去生成,最終拿到的都是這兩個(gè)接口的子類(lèi)的對(duì)象。

所以,對(duì)于 SqlSessionFactory 和 SqlSession 就沒(méi)法在 Spring 容器中直接進(jìn)行配置,那么對(duì)于這樣的 Bean,就可以通過(guò) xxxFactoryBean 來(lái)進(jìn)行配置。

我們來(lái)看下 SqlSessionFactoryBean 類(lèi),源碼很長(zhǎng),我挑了重要的出來(lái):

publicclassSqlSessionFactoryBeanimplementsFactoryBean,InitializingBean,ApplicationListener{

privateSqlSessionFactorysqlSessionFactory;

@Override
publicSqlSessionFactorygetObject()throwsException{
if(this.sqlSessionFactory==null){
afterPropertiesSet();
}

returnthis.sqlSessionFactory;
}
@Override
publicClassgetObjectType(){
returnthis.sqlSessionFactory==null?SqlSessionFactory.class:this.sqlSessionFactory.getClass();
}
@Override
publicbooleanisSingleton(){
returntrue;
}
}

大家看一下,SqlSessionFactoryBean 需要實(shí)現(xiàn) FactoryBean 接口,并且在實(shí)現(xiàn)接口的時(shí)候指定泛型是 SqlSessionFactory,也就是 SqlSessionFactoryBean 最終產(chǎn)出的 Bean 是 SqlSessionFactory。實(shí)現(xiàn)了 FactoryBean 接口之后,就需要實(shí)現(xiàn)接口中的三個(gè)方法:

getObject:這個(gè)方法返回的對(duì)象,就是真正要注冊(cè)到 Spring 容器中的對(duì)象,在這個(gè)方法中,我們就可以按照各種方式對(duì) Bean 進(jìn)行各種配置了。

getObjectType:這個(gè)方法返回注冊(cè)到 Spring 容器中的對(duì)象類(lèi)型。

isSingleton:這個(gè)方法用來(lái)返回注冊(cè)到 Spring 容器中的 Bean 是否是單例的。

這就是 FactoryBean 的特點(diǎn),由于某一個(gè) Bean 的初始化過(guò)于復(fù)雜,那么就可以通過(guò) FactoryBean 來(lái)幫助注冊(cè)到 Spring 容器中去。

2.2 實(shí)踐

松哥再寫(xiě)一個(gè)簡(jiǎn)單的例子給小伙伴們體驗(yàn)一把 FactoryBean。

假設(shè)我有如下類(lèi):

publicclassAuthor{

privateStringname;
privateIntegerage;

privateAuthor(){
}

publicstaticAuthorinit(Stringname,Integerage){
Authorauthor=newAuthor();
author.setAge(age);
author.setName(name);
returnauthor;
}
//省略getter/setter/toString
}

這個(gè)類(lèi)的特點(diǎn)就是構(gòu)造方法是私有的,你沒(méi)法從外面去 new,現(xiàn)在我想將這個(gè)類(lèi)的對(duì)象注冊(cè)到 Spring 容器中,那么我可以提供一個(gè) AuthorFactoryBean:

publicclassAuthorFactoryBeanimplementsFactoryBean{
@Override
publicAuthorgetObject()throwsException{
returnAuthor.init("javaboy",99);
}

@Override
publicClassgetObjectType(){
returnAuthor.class;
}

@Override
publicbooleanisSingleton(){
returntrue;
}
}

然后在 Spring 容器中配置 AuthorFactoryBean 即可:


接下來(lái)我們就可以從容器中去獲取 Author 對(duì)象了,但是要注意,通過(guò) author 這個(gè)名字拿到的是 Author 對(duì)象,而不是 AuthorFactoryBean 對(duì)象,如果想要獲取到 AuthorFactoryBean 對(duì)象,那么要通過(guò) &author 這個(gè)名字去獲取(回顧第一小節(jié)所講內(nèi)容)。

publicclassMain{
publicstaticvoidmain(String[]args){
ClassPathXmlApplicationContextctx=newClassPathXmlApplicationContext("applicationContext.xml");
Objectauthor=ctx.getBean("author");
ObjectauthorFactoryBean=ctx.getBean("&author");
System.out.println("author.getClass()="+author.getClass());
System.out.println("authorFactoryBean.getClass()="+authorFactoryBean.getClass());
}
}

來(lái)看下最終運(yùn)行結(jié)果:

e474637e-20a0-11ee-962d-dac502259ad0.png

跟我們所想的一致吧~

3. 小結(jié)

經(jīng)過(guò)前面的介紹,相信小伙伴們已經(jīng)能夠區(qū)分 BeanFactory 和 FactoryBean 了,再來(lái)回顧一下本文開(kāi)頭的內(nèi)容:

BeanFactory 是 Spring 框架的核心接口之一,用于管理和獲取應(yīng)用程序中的 Bean 實(shí)例。它是一個(gè)工廠模式的實(shí)現(xiàn),負(fù)責(zé)創(chuàng)建、配置和管理 Bean 對(duì)象。BeanFactory 是 Spring IoC 容器的基礎(chǔ),它可以從配置元數(shù)據(jù)(如 XML 文件)中讀取 Bean 的定義,并在需要時(shí)實(shí)例化和提供這些 Bean。

FactoryBean 是一個(gè)特殊的 Bean,它是一個(gè)工廠對(duì)象,用于創(chuàng)建和管理其他 Bean 的實(shí)例。FactoryBean 接口定義了一種創(chuàng)建 Bean 的方式,它允許開(kāi)發(fā)人員在 Bean 的創(chuàng)建過(guò)程中進(jìn)行更多的自定義操作。通過(guò)實(shí)現(xiàn) FactoryBean 接口,開(kāi)發(fā)人員可以創(chuàng)建復(fù)雜的 Bean 實(shí)例,或者在 Bean 實(shí)例化之前進(jìn)行一些額外的邏輯處理。

區(qū)別在于,BeanFactory 是 Spring 框架的核心接口,用于管理和提供 Bean 實(shí)例,而 FactoryBean 是一個(gè)特殊的 Bean,用于創(chuàng)建和管理其他 Bean 的實(shí)例。FactoryBean 在 Bean 的創(chuàng)建過(guò)程中提供更多的自定義能力,允許進(jìn)行額外的邏輯處理。






審核編輯:劉清

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 接觸器
    +關(guān)注

    關(guān)注

    63

    文章

    1196

    瀏覽量

    64337
  • XML技術(shù)
    +關(guān)注

    關(guān)注

    0

    文章

    15

    瀏覽量

    6011
  • IOC
    IOC
    +關(guān)注

    關(guān)注

    0

    文章

    28

    瀏覽量

    10099

原文標(biāo)題:Spring中BeanFactory和FactoryBean有何區(qū)別?

文章出處:【微信號(hào):OSC開(kāi)源社區(qū),微信公眾號(hào):OSC開(kāi)源社區(qū)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    樹(shù)莓派和香蕉派的區(qū)別在哪

    樹(shù)莓派和香蕉派到底有什么區(qū)別?國(guó)產(chǎn)還是英產(chǎn)的區(qū)別?就算山寨也有山寨的區(qū)別,如果不是山寨那到的區(qū)別在哪
    發(fā)表于 03-12 14:49

    模擬地與信號(hào)地的區(qū)別在哪里?

    模擬地與信號(hào)地的區(qū)別在哪里?在復(fù)雜混合信號(hào)PCB設(shè)計(jì)中有哪些注意事項(xiàng)?
    發(fā)表于 04-23 06:19

    TDD與FDD的區(qū)別在哪里?

    TDD與FDD的區(qū)別在哪里?TD-SCDMA系統(tǒng)優(yōu)勢(shì)有哪些?
    發(fā)表于 05-25 06:03

    string類(lèi)型數(shù)據(jù)與《string.h》頭文件的區(qū)別在哪里?

    string的含義是什么?string與vector容器的區(qū)別在哪?c的字符串是什么?string類(lèi)型數(shù)據(jù)與《string.h》頭文件的區(qū)別在哪里?
    發(fā)表于 07-05 07:18

    數(shù)字舵機(jī)與模擬舵機(jī)的區(qū)別在哪?

    舵機(jī)是什么?數(shù)字舵機(jī)的工作原理是什么?數(shù)字舵機(jī)與模擬舵機(jī)區(qū)別在哪
    發(fā)表于 07-13 06:16

    數(shù)字舵機(jī)與模擬舵機(jī)的區(qū)別在哪

    無(wú)刷舵機(jī)是由哪些部分組成的?數(shù)字舵機(jī)有何優(yōu)勢(shì)?數(shù)字舵機(jī)與模擬舵機(jī)的區(qū)別在哪
    發(fā)表于 08-10 06:00

    相電流和線電流的區(qū)別在哪

    什么是相電流?什么是線電流?相電流和線電流的區(qū)別在哪?
    發(fā)表于 09-29 07:44

    JTAG和SWD的區(qū)別在哪

    JTAG和SWD的區(qū)別在哪?USART和UART的區(qū)別在哪?
    發(fā)表于 10-08 09:01

    進(jìn)程是什么?進(jìn)程與程序的區(qū)別在哪

    進(jìn)程是什么?進(jìn)程與程序的區(qū)別在哪?進(jìn)程的狀態(tài)有哪幾種?
    發(fā)表于 12-23 06:27

    sizeof和strlen函數(shù)的區(qū)別在哪

    野指針是指指針指向的位置是不可知的,主要成因是什么?sizeof和strlen函數(shù)的區(qū)別在哪?鏈表和數(shù)組的區(qū)別在哪?
    發(fā)表于 12-24 07:19

    SoftMAC和FullMAC的區(qū)別在哪

    Wifi設(shè)備的協(xié)議棧是怎樣的?SoftMAC和FullMAC的區(qū)別在哪呢?SoftMAC和FullMAC分別有哪些優(yōu)勢(shì)呢?
    發(fā)表于 03-10 08:00

    RTK和GPS定位的區(qū)別在哪里?

    RTK和GPS定位的區(qū)別在哪里?
    發(fā)表于 05-08 10:08 ?76次下載

    串口屏和并口屏的區(qū)別在哪

    串口屏還是并口屏好用?區(qū)別在哪里?
    的頭像 發(fā)表于 01-23 09:53 ?9821次閱讀

    梯形絲桿和滾珠絲桿的區(qū)別在哪里?

    梯形絲桿和滾珠絲桿的區(qū)別在哪里?
    的頭像 發(fā)表于 03-28 17:48 ?2400次閱讀
    梯形絲桿和滾珠絲桿的<b class='flag-5'>區(qū)別在哪</b>里?

    BeanFactoryFactoryBean區(qū)別

    Spring框架BeanFactoryFactoryBean是兩個(gè)關(guān)鍵的概念,它們都與創(chuàng)建和管理Bean有關(guān),但它們?cè)诠δ芎妥饔蒙嫌泻艽蟮?b class='flag-5'>區(qū)
    的頭像 發(fā)表于 10-16 10:12 ?1077次閱讀
    RM新时代网站-首页