BeanFactory和FactoryBean的區(qū)別有哪些,針對這個問題,這篇文章詳細(xì)介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
創(chuàng)新互聯(lián)建站主營大觀網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營網(wǎng)站建設(shè)方案,成都App定制開發(fā),大觀h5小程序開發(fā)搭建,大觀網(wǎng)站營銷推廣歡迎大觀等地區(qū)企業(yè)咨詢
說實話,他倆除了名字比較像以外,好像沒有其他共同點了。
「BeanFactory和FactoryBean有哪些區(qū)別?」
BeanFactory是一個最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能
FactoryBean是創(chuàng)建Bean的一種方式,幫助實現(xiàn)復(fù)雜Bean的創(chuàng)建
和BeanFactory相關(guān)的還有一個高頻的面試題
「ApplicationContext和BeanFactory有哪些區(qū)別?」
BeanFactory是一個最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能
ApplicationContext繼承了BeanFactory,在BeanFactory的基礎(chǔ)上增加了企業(yè)級的功能,如AOP,資源管理(Resources)事件(Event),國際化(i18n),Environment抽象等
常見的創(chuàng)建Bean的方式有如下四種
通過構(gòu)造器
通過靜態(tài)工廠方法
通過Bean工廠方法
通過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; } }
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); } }
在分析源碼之前,我們先明確2個概念
「factoryBean是我們配置到容器中的實現(xiàn)FactoryBean接口的Bean,而subBean是用FactoryBean創(chuàng)建出來的Bean」
在Spring容器啟動的過程中,會實例化非延遲的單例Bean,即調(diào)用如下方法 DefaultListableBeanFactory#preInstantiateSingletons
調(diào)用FactoryBean#getObject的鏈路如下圖
通過分析DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject的調(diào)用鏈路可以分析得到
單例的factoryBean對象本身會在spring容器啟動時主動初始化。而subBean的初始化則是在第一次從緩存中獲取factoryBean并且不為空才會觸發(fā)
如果factoryBean對象實現(xiàn)的接口是SmartFactoryBean且isEagerInit方法返回true,那么subBean對象也會在spring容器啟動的時候主動初始化
如果bean注冊的時候,beanName對應(yīng)的bean實例是一個factoryBean,那么我們通過getBean(beanName)獲取到的對象將會是subBean對象;如果要獲取工廠對象factoryBean,需要使用getBean("&" + beanName)
單例的subBean也會緩存在spring容器中,具體的容器是FactoryBeanRegistrySupport#factoryBeanObjectCache,一個Map
「建議大家看一下DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject方法的調(diào)用鏈路,就能理解上面我說的流程了,我就不貼太多源碼了」
目前我只在Dubbbo源碼中看到了FactoryBean的應(yīng)用
「服務(wù)導(dǎo)出:在Dubbo中,服務(wù)提供者會被包裝成ServiceBean對象,當(dāng)監(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>
餓漢式:init=true,在Bean生命周期的初始化階段會調(diào)用InitializingBean#afterPropertiesSet方法,而這個方法會調(diào)用ReferenceBean#getObject方法,完成subBean的創(chuàng)建,即ReferenceBean實例化時完成服務(wù)引入
懶漢式: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 ReferenceBeanextends ReferenceConfig 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)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識。