日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

Spring5源码 - 08 BeanFactory和FactoryBean 源码解析 使用场景

發布時間:2025/3/21 javascript 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Spring5源码 - 08 BeanFactory和FactoryBean 源码解析 使用场景 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

文章目錄

  • BeanFactory VS FactoryBean
  • FactoryBean VS 普通Bean
  • 演示
  • 源碼
  • 使用場景


BeanFactory VS FactoryBean

首先明確一下,這兩個東西是完全不同的兩個東西 ,不要混淆。

BeanFactory 是Spring Framework的 頂級核心接口 , 沒有這個接口,就沒有Bean的產生。

FactoryBean也是一個接口,是一個特殊的Bean , 實現了FactoryBean 接口的Bean,原來的Bean將會被隱藏,而是由FactoryBean 的getObjecct方法返回最終的Bean 。 簡單工廠模式 。

如果需要獲取FactoryBean實例本身,需要 & 。

可以把FactoryBean理解為4S店,改裝你原來的Bean。


FactoryBean VS 普通Bean

FactoryBean和普通Bean不同,其返回的對象不是指定類的一個實例,而是該FactoryBean的getObject方法所返回的對象。創建出來的對象是否屬于單例由isSingleton中的返回決定。


演示


【Bean1】

package com.artisan.factoryBean;import org.springframework.stereotype.Component;import javax.annotation.PostConstruct;@Component public class Bean1 {@PostConstructpublic void init(){System.out.println("bean1 create ");} }

【SpecialBeanFB 實現了 FactoryBean接口 】 用于裝飾Bean

package com.artisan.factoryBean;import org.springframework.beans.factory.FactoryBean; import org.springframework.stereotype.Component;import javax.annotation.PostConstruct;@Component public class SpecialBeanFB implements FactoryBean {@PostConstructpublic void init(){System.out.println("SpecialBean as Factory Bean create ");}@Overridepublic Object getObject() {return new Bean2();}@Overridepublic Class<?> getObjectType() {return Bean2.class;}@Overridepublic boolean isSingleton() {return true;} }

如果SpecialBeanFB這個FactoryBean上的Component注解 增加了 name ,例如@Component("aaa") 其實是個getObject對象起的名字,而不是本身這個FactoryBean實例

當我們需要獲取FactoryBean實例本身而不是它所產生的bean,要使用&符號

比如這里的 id為”specialBeanFB”的FactoryBean :

  • 調用getBean(“specialBeanFB”)將返回FactoryBean產生的bean
  • 調用getBean("&specialBeanFB")將返回FactoryBean它本身的實例

【配置類】

package com.artisan.factoryBean;import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration;@Configuration @ComponentScan("com.artisan.factoryBean") public class FBConfig {}

【測試類】

package com.artisan.factoryBean;import org.springframework.context.annotation.AnnotationConfigApplicationContext;public class FactoryBeanTest {public static void main(String[] args) throws Exception {// 實例化Spring Bean 容器AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(FBConfig.class);// 從bean容器中讀取普通的beanSystem.out.println(ac.getBean(Bean1.class));System.out.println(ac.getBean("bean1"));System.out.println("===========");// 從bean容器中讀取FactoryBean接口修飾的BeanSystem.out.println(ac.getBean(SpecialBeanFB.class).getObject().getClass().getName());System.out.println(ac.getBean("specialBeanFB"));System.out.println("===========");// 通過 & 獲取 FactoryBean本身的bean對象System.out.println(ac.getBean("&specialBeanFB"));System.out.println("===========");// 如果SpecialBeanFB這個FactoryBean上的Component注解 增加了 name ,例如@Component("aaa")// 其實是個getObject對象起的名字,而不是本身這個FactoryBean實例 // System.out.println(ac.getBean("aaa")); // System.out.println(ac.getBean("&aaa"));} }

【測試結果】


源碼

在實例化Bean的方法中

AbstractApplicationContext # refresh() ---------- finishBeanFactoryInitialization(beanFactory) ----- DefaultListableBeanFactory#preInstantiateSingletons

在Spring容器啟動階段,會調用到refresh()方法,在refresh()中調用了finishBeanFactoryInitialization()方法,最終會調用到beanFactory.preInstantiateSingletons()方法

public void preInstantiateSingletons() throws BeansException {// 從容器中獲取到所有的beanNameList<String> beanNames = new ArrayList<>(this.beanDefinitionNames);for (String beanName : beanNames) {// 合并Bean ScannedGenericBeanDefinition AnnotatedGenericBeanDefinition 類型的Bean 等等 都要合并為 RootBeanDefinition 比較復雜,知道即可RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);// 創建Bean的條件校驗if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {// 在此處會根據beanName判斷bean是不是一個FactoryBean,實現了FactoryBean接口的bean,會返回true if (isFactoryBean(beanName)) {// 然后通過getBean()方法去獲取或者創建單例對象// 注意:在此處為beanName拼接了一個前綴:FACTORY_BEAN_PREFIX 是一個常量字符串,即:&// 所以在此時容器啟動階段,對于specialBeanFB,應該是:getBean("&specialBeanFB")Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);// 下面這一段邏輯,是判斷是否需要在容器啟動階段,就去實例化getObject()返回的對象,即是否調用FactoryBean的getObject()方法if (bean instanceof FactoryBean) {final FactoryBean<?> factory = (FactoryBean<?>) bean;boolean isEagerInit;if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) {isEagerInit = AccessController.doPrivileged((PrivilegedAction<Boolean>)((SmartFactoryBean<?>) factory)::isEagerInit,getAccessControlContext());}else {isEagerInit = (factory instanceof SmartFactoryBean &&((SmartFactoryBean<?>) factory).isEagerInit());}if (isEagerInit) {getBean(beanName);}}}}} }

使用場景

比如 MyBatis3 提供 mybatis-spring項目中的 org.mybatis.spring.SqlSessionFactoryBean

SqlSessionFactoryBean 看名字的結尾FactoryBean

public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory>, InitializingBean, ApplicationListener<ApplicationEvent> {// ...省略其他代碼public SqlSessionFactory getObject() throws Exception {if (this.sqlSessionFactory == null) {afterPropertiesSet();}return this.sqlSessionFactory;} }

sqlSessionFactory是SqlSessionFactoryBean的一個屬性,它的賦值是在通過回調afterPropertiesSet()方法進行的。 因為SqlSessionFactoryBean實現了InitializingBean接口,所以在Spring初始化Bean的時候,能回調afterPropertiesSet()方法 .

public void afterPropertiesSet() throws Exception {// buildSqlSessionFactory()方法會根據mybatis的配置進行初始化。this.sqlSessionFactory = buildSqlSessionFactory(); }

總結

以上是生活随笔為你收集整理的Spring5源码 - 08 BeanFactory和FactoryBean 源码解析 使用场景的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。