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