溫馨提示×

溫馨提示×

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

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

BeanFactory和FactoryBean的區(qū)別有哪些

發(fā)布時間:2021-09-13 14:52:54 來源:億速云 閱讀:179 作者:柒染 欄目:web開發(fā)

BeanFactory和FactoryBean的區(qū)別有哪些,針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

BeanFactory和FactoryBean的區(qū)別有哪些

區(qū)別

說實話,他倆除了名字比較像以外,好像沒有其他共同點了。

「BeanFactory和FactoryBean有哪些區(qū)別?」

  1. BeanFactory是一個最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能

  2. FactoryBean是創(chuàng)建Bean的一種方式,幫助實現(xiàn)復(fù)雜Bean的創(chuàng)建

和BeanFactory相關(guān)的還有一個高頻的面試題

「ApplicationContext和BeanFactory有哪些區(qū)別?」

  1. BeanFactory是一個最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能

  2. ApplicationContext繼承了BeanFactory,在BeanFactory的基礎(chǔ)上增加了企業(yè)級的功能,如AOP,資源管理(Resources)事件(Event),國際化(i18n),Environment抽象等

創(chuàng)建Bean的方式

常見的創(chuàng)建Bean的方式有如下四種

  1. 通過構(gòu)造器

  2. 通過靜態(tài)工廠方法

  3. 通過Bean工廠方法

  4. 通過FactoryBean

@Data @ToString public class User {   private Long id;  private String name;   public static User createUser() {   User user = new User();   user.setId(1L);   user.setName("li");   return user;  } }
public class UserFactory {   public User createUser() {   return User.createUser();  } }
public class UserFactoryBean implements FactoryBean {   @Override  public Object getObject() throws Exception {   return User.createUser();  }   @Override  public Class<?> getObjectType() {   return User.class;  } }
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans"     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"     xsi:schemaLocation="http://www.springframework.org/schema/beans         https://www.springframework.org/schema/beans/spring-beans.xsd">   <!-- 構(gòu)造方法實例化 Bean -->  <bean id="user-by-constructor" class="com.javashitang.domain.User">   <property name="id" value="1"/>         <property name="name" value="li"/>  </bean>   <!-- 靜態(tài)方法實例化 Bean -->  <bean id="user-by-static-method" class="com.javashitang.domain.User"     factory-method="createUser"/>   <bean id="userFactory" class="com.javashitang.factory.UserFactory"/>   <!-- Bean工廠方法實例化 Bean -->  <bean id="user-by-factory" factory-bean="userFactory" factory-method="createUser"/>   <!-- FactoryBean實例化 Bean -->  <bean id="user-by-factory-bean" class="com.javashitang.factory.UserFactoryBean"/> </beans>
public class BeanInstantiationDemo {   public static void main(String[] args) {   BeanFactory beanFactory = new ClassPathXmlApplicationContext("classpath:/bean-instantiation-context.xml");   User user1 = beanFactory.getBean("user-by-constructor", User.class);   User user2 = beanFactory.getBean("user-by-static-method", User.class);   User user3 = beanFactory.getBean("user-by-factory", User.class);   User user4 = beanFactory.getBean("user-by-factory-bean", User.class);  } }

實現(xiàn)原理

在分析源碼之前,我們先明確2個概念

「factoryBean是我們配置到容器中的實現(xiàn)FactoryBean接口的Bean,而subBean是用FactoryBean創(chuàng)建出來的Bean」

在Spring容器啟動的過程中,會實例化非延遲的單例Bean,即調(diào)用如下方法  DefaultListableBeanFactory#preInstantiateSingletons

BeanFactory和FactoryBean的區(qū)別有哪些

調(diào)用FactoryBean#getObject的鏈路如下圖

BeanFactory和FactoryBean的區(qū)別有哪些

通過分析DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject的調(diào)用鏈路可以分析得到

  1. 單例的factoryBean對象本身會在spring容器啟動時主動初始化。而subBean的初始化則是在第一次從緩存中獲取factoryBean并且不為空才會觸發(fā)

  2. 如果factoryBean對象實現(xiàn)的接口是SmartFactoryBean且isEagerInit方法返回true,那么subBean對象也會在spring容器啟動的時候主動初始化

  3. 如果bean注冊的時候,beanName對應(yīng)的bean實例是一個factoryBean,那么我們通過getBean(beanName)獲取到的對象將會是subBean對象;如果要獲取工廠對象factoryBean,需要使用getBean("&"  + beanName)

  4. 單例的subBean也會緩存在spring容器中,具體的容器是FactoryBeanRegistrySupport#factoryBeanObjectCache,一個Map

「建議大家看一下DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject方法的調(diào)用鏈路,就能理解上面我說的流程了,我就不貼太多源碼了」

應(yīng)用

目前我只在Dubbbo源碼中看到了FactoryBean的應(yīng)用

「服務(wù)導(dǎo)出:在Dubbo中,服務(wù)提供者會被包裝成ServiceBean對象,當監(jiān)聽到ContextRefreshedEvent事件時開始服務(wù)導(dǎo)出」

「服務(wù)調(diào)用:服務(wù)調(diào)用方會被包裝成ReferenceBean對象,ReferenceBean實現(xiàn)了FactoryBean接口和InitializingBean接口,創(chuàng)建subBean的邏輯在ReferenceBean#getObject方法中」

「Dubbo服務(wù)引入的時機有如下2種?!?/p>

  1. 餓漢式:init=true,在Bean生命周期的初始化階段會調(diào)用InitializingBean#afterPropertiesSet方法,而這個方法會調(diào)用ReferenceBean#getObject方法,完成subBean的創(chuàng)建,即ReferenceBean實例化時完成服務(wù)引入

  2. 懶漢式:init=false,在ReferenceBean對應(yīng)的服務(wù)被注入到其他類中時,此時會調(diào)用AbstractApplicationContext#getBean,獲取ReferenceBean對象,因為ReferenceBean實現(xiàn)了FactoryBean接口,所以會調(diào)用ReferenceBean#getObject方法,完成subBean的創(chuàng)建,即完成服務(wù)引入

public class ReferenceBean<T> extends ReferenceConfig<T> implements FactoryBean, ApplicationContextAware, InitializingBean, DisposableBean {      @Override     public Object getObject() {         return get();     }      @Override     @SuppressWarnings({"unchecked"})     public void afterPropertiesSet() throws Exception {          // 省略部分代碼          if (shouldInit()) {             getObject();         }     }      }

關(guān)于BeanFactory和FactoryBean的區(qū)別有哪些問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。

向AI問一下細節(jié)

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

AI