面試官:BeanFactory和FactoryBean有哪些區(qū)別?
區(qū)別
說實(shí)話,他倆除了名字比較像以外,好像沒有其他共同點(diǎn)了。
「BeanFactory和FactoryBean有哪些區(qū)別?」
- BeanFactory是一個(gè)最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能
- FactoryBean是創(chuàng)建Bean的一種方式,幫助實(shí)現(xiàn)復(fù)雜Bean的創(chuàng)建
和BeanFactory相關(guān)的還有一個(gè)高頻的面試題
「ApplicationContext和BeanFactory有哪些區(qū)別?」
- BeanFactory是一個(gè)最基礎(chǔ)的IOC容器,提供了依賴查找,依賴注入等基礎(chǔ)的功能
- ApplicationContext繼承了BeanFactory,在BeanFactory的基礎(chǔ)上增加了企業(yè)級(jí)的功能,如AOP,資源管理(Resources)事件(Event),國(guó)際化(i18n),Environment抽象等
創(chuàng)建Bean的方式
常見的創(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;
- }
- }
- <?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)造方法實(shí)例化 Bean -->
- <bean id="user-by-constructor" class="com.javashitang.domain.User">
- <property name="id" value="1"/>
- <property name="name" value="li"/>
- </bean>
- <!-- 靜態(tài)方法實(shí)例化 Bean -->
- <bean id="user-by-static-method" class="com.javashitang.domain.User"
- factory-method="createUser"/>
- <bean id="userFactory" class="com.javashitang.factory.UserFactory"/>
- <!-- Bean工廠方法實(shí)例化 Bean -->
- <bean id="user-by-factory" factory-bean="userFactory" factory-method="createUser"/>
- <!-- FactoryBean實(shí)例化 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);
- }
- }
實(shí)現(xiàn)原理
在分析源碼之前,我們先明確2個(gè)概念
「factoryBean是我們配置到容器中的實(shí)現(xiàn)FactoryBean接口的Bean,而subBean是用FactoryBean創(chuàng)建出來的Bean」
在Spring容器啟動(dòng)的過程中,會(huì)實(shí)例化非延遲的單例Bean,即調(diào)用如下方法 DefaultListableBeanFactory#preInstantiateSingletons
調(diào)用FactoryBean#getObject的鏈路如下圖
通過分析DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject的調(diào)用鏈路可以分析得到
- 單例的factoryBean對(duì)象本身會(huì)在spring容器啟動(dòng)時(shí)主動(dòng)初始化。而subBean的初始化則是在第一次從緩存中獲取factoryBean并且不為空才會(huì)觸發(fā)
- 如果factoryBean對(duì)象實(shí)現(xiàn)的接口是SmartFactoryBean且isEagerInit方法返回true,那么subBean對(duì)象也會(huì)在spring容器啟動(dòng)的時(shí)候主動(dòng)初始化
- 如果bean注冊(cè)的時(shí)候,beanName對(duì)應(yīng)的bean實(shí)例是一個(gè)factoryBean,那么我們通過getBean(beanName)獲取到的對(duì)象將會(huì)是subBean對(duì)象;如果要獲取工廠對(duì)象factoryBean,需要使用getBean("&" + beanName)
- 單例的subBean也會(huì)緩存在spring容器中,具體的容器是FactoryBeanRegistrySupport#factoryBeanObjectCache,一個(gè)Map
「建議大家看一下DefaultListableBeanFactory#preInstantiateSingletons方法和FactoryBean#getObject方法的調(diào)用鏈路,就能理解上面我說的流程了,我就不貼太多源碼了」
應(yīng)用
目前我只在Dubbbo源碼中看到了FactoryBean的應(yīng)用
「服務(wù)導(dǎo)出:在Dubbo中,服務(wù)提供者會(huì)被包裝成ServiceBean對(duì)象,當(dāng)監(jiān)聽到ContextRefreshedEvent事件時(shí)開始服務(wù)導(dǎo)出」
「服務(wù)調(diào)用:服務(wù)調(diào)用方會(huì)被包裝成ReferenceBean對(duì)象,ReferenceBean實(shí)現(xiàn)了FactoryBean接口和InitializingBean接口,創(chuàng)建subBean的邏輯在ReferenceBean#getObject方法中」
「Dubbo服務(wù)引入的時(shí)機(jī)有如下2種?!?/p>
- 餓漢式:init=true,在Bean生命周期的初始化階段會(huì)調(diào)用InitializingBean#afterPropertiesSet方法,而這個(gè)方法會(huì)調(diào)用ReferenceBean#getObject方法,完成subBean的創(chuàng)建,即ReferenceBean實(shí)例化時(shí)完成服務(wù)引入
- 懶漢式:init=false,在ReferenceBean對(duì)應(yīng)的服務(wù)被注入到其他類中時(shí),此時(shí)會(huì)調(diào)用AbstractApplicationContext#getBean,獲取ReferenceBean對(duì)象,因?yàn)镽eferenceBean實(shí)現(xiàn)了FactoryBean接口,所以會(huì)調(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();
- }
- }
- }
本文轉(zhuǎn)載自微信公眾號(hào)「Java識(shí)堂」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系Java識(shí)堂公眾號(hào)。