溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何理解Spring中BeanFactory與FactoryBean的區(qū)別

發(fā)布時(shí)間:2021-10-15 17:46:15 來(lái)源:億速云 閱讀:113 作者:柒染 欄目:編程語(yǔ)言

這篇文章將為大家詳細(xì)講解有關(guān)如何理解Spring中BeanFactory與FactoryBean的區(qū)別,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。

在Spring中有BeanFactory和FactoryBean這2個(gè)接口,從名字來(lái)看很相似,比較容易搞混。

一、BeanFactory

BeanFactory是一個(gè)接口,它是Spring中工廠的頂層規(guī)范,是SpringIoc容器的核心接口,它定義了getBean()、containsBean()等管理Bean的通用方法。Spring的容器都是它的具體實(shí)現(xiàn)如:

DefaultListableBeanFactory  XmlBeanFactory  ApplicationContext

這些實(shí)現(xiàn)類又從不同的維度分別有不同的擴(kuò)展。

1.1 BenaFactory源碼

public interface BeanFactory {  /**   * 用來(lái)獲得實(shí)例的引用,并且區(qū)分FactoryBean區(qū)分。   * 如果使用bean的名字myJndiObject獲取FactoryBean,返回的是一個(gè)工廠,而不是工廠的實(shí)例;如果需要獲得工廠實(shí)例,需要轉(zhuǎn)義。   */  String FACTORY_BEAN_PREFIX = "&";  /**   * 根據(jù)bean的名稱,獲取指定的bean實(shí)例,該實(shí)例可以是共享的,也可以是獨(dú)立的.   */  Object getBean(String name) throws BeansException;  /**   * 根據(jù)bean的名稱,獲取指定的bean實(shí)例,該實(shí)例可以是共享的,也可以是獨(dú)立的.并且增加了一個(gè)類型的檢驗(yàn)。   */  <T> T getBean(String name, Class<T> requiredType) throws BeansException;  Object getBean(String name, Object... args) throws BeansException;  /**   * 根據(jù)給定類型返回匹配的bean實(shí)例.   */  <T> T getBean(Class<T> requiredType) throws BeansException;  <T> T getBean(Class<T> requiredType, Object... args) throws BeansException;  /**   * 檢查spring的bean容器中是否包含有該bean   */  boolean containsBean(String name);  /**   * 判斷bean的作用域是否是singleton   */  boolean isSingleton(String name) throws NoSuchBeanDefinitionException;  /**   * 判斷bena的作用域是否是prototype   */  boolean isPrototype(String name) throws NoSuchBeanDefinitionException;  /**   * 檢查給定名稱的bean是否和指定類型匹配.更確卻的說(shuō)是通過(guò)檢查給定的bean,返回指定類型的目標(biāo)對(duì)象   */  boolean isTypeMatch(String name, ResolvableType typeToMatch) throws NoSuchBeanDefinitionException;  boolean isTypeMatch(String name, Class<?> typeToMatch) throws NoSuchBeanDefinitionException;  /**   * 獲取給定名稱的bean的class類型   */  Class<?> getType(String name) throws NoSuchBeanDefinitionException;  /**   * 獲取給定bean名稱的別名,如果根據(jù)別名檢索,將會(huì)獲得原始bean名稱。   *   String[] getAliases(String name);}

1.2 使用場(chǎng)景

從Ioc容器中獲取Bean(byName or byType):context.getBean("father", Father.class)、context.getBean("father")  檢索Ioc容器中是否包含指定的Bean: context.containsBean("father")  判斷Bean是否為單例: context.isSingleton("father")

二、FactoryBean

首先它是一個(gè)Bean,但又不僅僅是一個(gè)Bean。它是一個(gè)能生產(chǎn)或修飾對(duì)象生成的工廠Bean,類似于設(shè)計(jì)模式中的工廠模式和裝飾器模式。它能在需要的時(shí)候生產(chǎn)一個(gè)對(duì)象,且不僅僅限于它自身,它能返回任何Bean的實(shí)例。

2.1 FactoryBean源碼

public interface FactoryBean<T> {  /**   * 從工廠中獲取bean實(shí)例   */  T getObject() throws Exception;  /**   * 從工廠中獲取bean實(shí)例對(duì)象的類型   */  Class<?> getObjectType();  /**   * 工廠創(chuàng)建的對(duì)象是否是單例   */  boolean isSingleton();}

從它定義的接口可以看出,F(xiàn)actoryBean表現(xiàn)的是一個(gè)工廠的職責(zé)。 即一個(gè)Bean A如果實(shí)現(xiàn)了FactoryBean接口,那么A就變成了一個(gè)工廠,根據(jù)A的名稱獲取到的實(shí)際上是工廠調(diào)用getObject()返回的對(duì)象,而不是A本身,如果要獲取工廠A自身的實(shí)例,那么需要在名稱前面加上'&'符號(hào)。

getObject('name')返回工廠中的實(shí)例  getObject('&name')返回工廠本身的實(shí)例

通常情況下,bean 無(wú)須自己實(shí)現(xiàn)工廠模式,Spring 容器擔(dān)任了工廠的 角色;但少數(shù)情況下,容器中的 bean 本身就是工廠,作用是產(chǎn)生其他 bean 實(shí)例。由工廠 bean 產(chǎn)生的其他 bean 實(shí)例,不再由 Spring 容器產(chǎn)生,因此與普通 bean 的配置不同,不再需要提供 class 元素。

2.2 示例

先定義一個(gè)Bean實(shí)現(xiàn)FactoryBean接口:

@Componentpublic class MyBean implements FactoryBean {  private String message;  public MyBean() {    this.message = "通過(guò)構(gòu)造方法初始化實(shí)例";  }  public MyBean(String message) {    this.message = message;  }  @Override  public Object getObject() throws Exception {    // 這里并不一定要返回MyBean自身的實(shí)例,可以是其他任何對(duì)象的實(shí)例    return new MyBean("通過(guò)FactoryBean.getObject()創(chuàng)建實(shí)例");  }  @Override  public Class<?> getObjectType() {    return MyBean.class;  }  public String getMessage() {    return message;  }  @Override  public boolean isSingleton() {    return false;  }}

MyBean實(shí)現(xiàn)了FactoryBean接口的三個(gè)方法,getObject()是可以返回任何對(duì)象的實(shí)例的,這里測(cè)試就返回MyBean自身實(shí)例,且返回前給message字段賦值。同時(shí)在構(gòu)造方法中也為message賦值。然后測(cè)試代碼中先通過(guò)名稱獲取Bean實(shí)例,打印message的內(nèi)容,再通過(guò)&+名稱獲取實(shí)例并打印message內(nèi)容。

@RunWith(SpringRunner.class)@SpringBootTest(classes = HelloApplication.class)public class FactoryBeanTest {  @Autowired  private ApplicationContext context;  @Test  public void test() {    MyBean myBean1 = (MyBean) context.getBean("myBean");//返回工廠中的實(shí)例,調(diào)用FactoryBean.getObject()創(chuàng)建實(shí)例    System.out.println("myBean1 = " + myBean1.getMessage());    MyBean myBean2 = (MyBean) context.getBean("&myBean");//返回工廠本身,通過(guò)構(gòu)造方法初始化實(shí)例    System.out.println("myBean2 = " + myBean2.getMessage());    System.out.println("myBean1.equals(myBean2) = " + myBean1.equals(myBean2));  }}

打印結(jié)果:

myBean1 = 通過(guò)FactoryBean.getObject()創(chuàng)建實(shí)例myBean2 = 通過(guò)構(gòu)造方法初始化實(shí)例myBean1.equals(myBean2) = false

2.3 使用場(chǎng)景

說(shuō)了這么多,為什么要有FactoryBean這個(gè)東西呢,有什么具體的作用嗎?

FactoryBean在Spring中最為典型的一個(gè)應(yīng)用就是用來(lái)創(chuàng)建AOP的代理對(duì)象。

我們知道AOP實(shí)際上是Spring在運(yùn)行時(shí)創(chuàng)建了一個(gè)代理對(duì)象,也就是說(shuō)這個(gè)對(duì)象,是我們?cè)谶\(yùn)行時(shí)創(chuàng)建的,而不是一開始就定義好的,這很符合工廠方法模式。更形象地說(shuō),AOP代理對(duì)象通過(guò)Java的反射機(jī)制,在運(yùn)行時(shí)創(chuàng)建了一個(gè)代理對(duì)象,在代理對(duì)象的目標(biāo)方法中根據(jù)業(yè)務(wù)要求織入了相應(yīng)的方法。這個(gè)對(duì)象在Spring中就是——ProxyFactoryBean。

所以,F(xiàn)actoryBean為我們實(shí)例化Bean提供了一個(gè)更為靈活的方式,我們可以通過(guò)FactoryBean創(chuàng)建出更為復(fù)雜的Bean實(shí)例。

三、區(qū)別

他們兩個(gè)都是個(gè)工廠,但FactoryBean本質(zhì)上還是一個(gè)Bean,也歸BeanFactory管理  BeanFactory是Spring容器的頂層接口,F(xiàn)actoryBean更類似于用戶自定義的工廠接口。

關(guān)于如何理解Spring中BeanFactory與FactoryBean的區(qū)別就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI