BeanFactory體繫結構

来源:https://www.cnblogs.com/zhangfengxian/archive/2019/08/03/11296591.html
-Advertisement-
Play Games

BeanFactory是Spring中非常重要的一個類,搞懂了它,你就知道了bean的初始化和摧毀過程,對於深入理解IOC有很大的幫助。 BeanFactory體繫結構 首先看一下使用IDEA生成的繼承層次圖(圖中去掉了ApplicationContext的繼承圖): 可以看到 下的介面主要分為三個 ...


BeanFactory是Spring中非常重要的一個類,搞懂了它,你就知道了bean的初始化和摧毀過程,對於深入理解IOC有很大的幫助。

BeanFactory體繫結構

首先看一下使用IDEA生成的繼承層次圖(圖中去掉了ApplicationContext的繼承圖):

可以看到BeanFactory下的介面主要分為三個:

關於BeanFactory的分析見https://www.cnblogs.com/zhangfengxian/p/11086695.html#beanfactory

BeanFactory介面分析

下麵將對BeanFactory中的介面進行分析。

AutowireCapableBeanFactory

該介面提供了對現有bean進行自動裝配的能力,設計目的不是為了用於一般的應用代碼中,對於一般的應用代碼應該使用BeanFactoryListableBeanFactory。其他框架的代碼集成可以利用這個介面去裝配和填充現有的bean的實例,但是Spring不會控制這些現有bean的生命周期。你也許註意到了ApplicationContext中的getAutowireCapableBeanFactory()能獲取到AutowireCapableBeanFactory的實例(https://www.cnblogs.com/zhangfengxian/p/11086695.html#applicationcontext%E8%AE%BE%E8%AE%A1%E8%A7%A3%E6%9E%90)。同樣,也能實現BeanFactoryAware介面來接收BeanFactory(應用程式上下暴露的內部使用的BeanFactory)的實例,然後將其轉換成AutowireCapableBeanFactory

下麵看一下這個介面中的靜態成員變數和方法:

// 定義了bean的裝配策略
int AUTOWIRE_NO = 0;            // 不進行裝配
int AUTOWIRE_BY_NAME = 1;       // 根據名字進行裝配
int AUTOWIRE_BY_TYPE = 2;       // 根據類型進行裝配
int AUTOWIRE_CONSTRUCTOR = 3;   // 根據構造函數進行裝配
@Deprecated
int AUTOWIRE_AUTODETECT = 4;    // Spring3.0已經過時的方法,通過省視bean來決定適當的裝載策略

//Spring5.1後增加,原始實例的尾碼,例如"com.mypackage.MyClass.ORIGINAL",強制返回給定的實例(沒有代理)
String ORIGINAL_INSTANCE_SUFFIX = ".ORIGINAL";

// 完全創建給定類的一個新的實例,包括所有適用的BeanPostProcessor
// 填充註解的field和方法,並且會應用所有的初始化回調函數
<T> T createBean(Class<T> beanClass) throws BeansException;
// 裝配bean,通過應用初始化之後的回調函數和bean屬性的後置處理來填充給定的bean的實例
// 本質上是為了在創建新的實例或者反序列化實例時,填充(重新填充)實例中註解的field和方法
void autowireBean(Object existingBean) throws BeansException;
// 配置給定的原始bean:自動裝配bean的屬性,應用bean的屬性值、工廠回調函數(例如setBeanName,values)
// 同樣也會應用所有bean的後置處理器
Object configureBean(Object existingBean, String beanName) throws BeansException;

// 使用指定的裝配策略來完全創建一個新的bean的實例
Object createBean(Class<?> beanClass, int autowireMode, boolean dependencyCheck) throws BeansException;
// 使用指定的裝配策略來實例化一個給定類新的bean的實例
// 不會應用標準的BeanPostProcessor回調函數或者在未來執行任何bean的初始化
Object autowire(Class<?> beanClass, int autowireMode, boolean dependencyCheck) throws BeansException;
// 根據名字和類型來自動裝配給定bean的實例的屬性
// 不會應用標準的BeanPostProcessor回調函數或者在未來執行任何bean的初始化
void autowireBeanProperties(Object existingBean, int autowireMode, boolean dependencyCheck)
    throws BeansException;
// 應用給定名字的bean的定義的屬性值到給定的bean的實例
// 該方法不會自動裝配bean屬性,僅僅應用明確定義的屬性值
void applyBeanPropertyValues(Object existingBean, String beanName) throws BeansException;
// 初始化給定的原始的bean應用bean的屬性值、工廠回調函數(例如setBeanName,values)
// 同樣也會應用所有bean的後置處理器
Object initializeBean(Object existingBean, String beanName) throws BeansException;
// 應用BeanPostProcessor到給定的現存的bean的實例,調用postProcessBeforeInitialization方法
Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName)
    throws BeansException;
// 應用BeanPostProcessor到給定的現存的bean的實例,postProcessAfterInitialization
Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
    throws BeansException;
// 摧毀給定的bean的實例,應用DisposableBean規約和註冊的DestructionAwareBeanPostProcessor
void destroyBean(Object existingBean);

// 解析唯一匹配給定對象類型的bean的實例,該方法是getBean(Class)的變種,只不過它還提供了匹配實例的bean的名字
<T> NamedBeanHolder<T> resolveNamedBean(Class<T> requiredType) throws BeansException;
// 解析給定bean的名字的實例,提供了用於暴露目標的工廠方法的依賴描述符
Object resolveBeanByName(String name, DependencyDescriptor descriptor) throws BeansException;
// 針對在工廠中定義的bean來解析指定的依賴
Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName) throws BeansException;
// 針對在工廠中定義的bean來解析指定的依賴
Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName,
                         @Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException;

ConfigurableBeanFactory

ConfigurableBeanFactory提供了bean工廠的配置機制(除了BeanFactory介面中的bean的工廠的客戶端方法)。該BeanFactory介面不適應一般的應用代碼中,應該使用BeanFactoryListableBeanFactory。該擴展介面僅僅用於內部框架的使用,並且是對bean工廠配置方法的特殊訪問。

ConfigurableBeanFactory繼承自HierarchicalBeanFactorySingletonBeanRegistry,下麵先看下SingletonBeanRegistry

SingletonBeanRegistry是為了共用的bean的實例而定義的註冊器,以統一的方式暴露單例管理機制。下麵是在此介面中定義的方法:

// 在bean的註冊器中以給定的bean的名字將給定的現存對象註冊為單例
void registerSingleton(String beanName, Object singletonObject);
// 根據給定的bean的名字來獲取單例bean,可能為null
Object getSingleton(String beanName);
// 是否包含給定名字的單例bean
boolean containsSingleton(String beanName);
// 獲取所有在註冊器中註冊的單例bean的名字
String[] getSingletonNames();
// 獲取所有在註冊器中註冊的單例bean的數量
int getSingletonCount();
// 獲取在這個註冊器中使用的單例的mutex(用於外部協同)
Object getSingletonMutex();

需要註意的是使用registerSingleton方法註冊的單例bean,不會執行任何的初始化回調函數(尤其不會調用InitializingBeanafterPropertiesSet方法),同樣也不會接收任何的摧毀回調函數。如果需要接收初始化和摧毀回調函數,請註冊bean的定義而不是現存的實例對象。

接下來看下ConfigurableBeanFactory中定義的方法:

// 作用域
String SCOPE_SINGLETON = "singleton"; // 單例作用域
String SCOPE_PROTOTYPE = "prototype"; // 原型作用域

// 設置父級bean工廠
void setParentBeanFactory(BeanFactory parentBeanFactory) throws IllegalStateException;
// 設置bean的類載入器,預設為線程上下文類載入器
void setBeanClassLoader(@Nullable ClassLoader beanClassLoader);
// 獲取bean的類載入器
@Nullable
ClassLoader getBeanClassLoader();
// 設置臨時的類載入器
void setTempClassLoader(@Nullable ClassLoader tempClassLoader);
// 獲取臨時的類載入器
@Nullable
ClassLoader getTempClassLoader();
// 設置是否緩存bean的元數據
void setCacheBeanMetadata(boolean cacheBeanMetadata);
// 是否緩存bean的元數據
boolean isCacheBeanMetadata();
// 設置bean的表達式解析器,以統一的EL相容樣式支持#{...}這樣的表達式
void setBeanExpressionResolver(@Nullable BeanExpressionResolver resolver);
// 獲取bean的表達式解析器
@Nullable
BeanExpressionResolver getBeanExpressionResolver();
// 設置轉換服務,用於轉換屬性值
void setConversionService(@Nullable ConversionService conversionService);
// 獲取轉換服務
@Nullable
ConversionService getConversionService();
// 添加屬性編輯器註冊者
void addPropertyEditorRegistrar(PropertyEditorRegistrar registrar);
// 為所有給定的屬性註冊自定義屬性編輯器
void registerCustomEditor(Class<?> requiredType, Class<? extends PropertyEditor> propertyEditorClass);
// 使用在BeanFactory中註冊的自定義編輯器來初始哈給定的屬性編輯器註冊者
void copyRegisteredEditorsTo(PropertyEditorRegistry registry);
// 設置類型轉換器
void setTypeConverter(TypeConverter typeConverter);
// 獲取類型轉換器
TypeConverter getTypeConverter();
// 添加嵌入值解析器,例如註冊屬性
void addEmbeddedValueResolver(StringValueResolver valueResolver);
// 在BeanFactory是否有註冊嵌入值解析器
boolean hasEmbeddedValueResolver();
// 解析給定的嵌入的值
@Nullable
String resolveEmbeddedValue(String value);
// 添加bean的後置處理器
void addBeanPostProcessor(BeanPostProcessor beanPostProcessor);
// 獲取bean的後置處理器個數
int getBeanPostProcessorCount();
// 註冊作用域
void registerScope(String scopeName, Scope scope);
// 獲取註冊的作用域的名字
String[] getRegisteredScopeNames();
// 獲取作用域
@Nullable
Scope getRegisteredScope(String scopeName);
// 提供一個與這個工廠有關的安全訪問控制上下文
AccessControlContext getAccessControlContext();
// 從給定的其他的工廠拷貝所有相關的配置。不應該包含任何bean的定義元數據
void copyConfigurationFrom(ConfigurableBeanFactory otherFactory);
// 註冊別名
void registerAlias(String beanName, String alias) throws BeanDefinitionStoreException;
// 解析所有別名的目標名稱和在工廠中註冊的別名,將給定的StringValueResolver應用於它們
void resolveAliases(StringValueResolver valueResolver);
// 獲取合併的bean的定義
BeanDefinition getMergedBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 給定名字的bean是否為FactoryBean
boolean isFactoryBean(String name) throws NoSuchBeanDefinitionException;
// 顯式的設置指定bean的目前在創建狀態
void setCurrentlyInCreation(String beanName, boolean inCreation);
// 指定的bean目前是否為在建狀態
boolean isCurrentlyInCreation(String beanName);
// 註冊給定bean所依賴的bean
void registerDependentBean(String beanName, String dependentBeanName);
// 獲取所有依賴於指定bean的bean的名字
String[] getDependentBeans(String beanName);
// 獲取所有指定bean所依賴的bean的名字
String[] getDependenciesForBean(String beanName);
// 根據bean的定義來摧毀給定的bean的實例(通常是從工廠中獲取到的原型實例)
void destroyBean(String beanName, Object beanInstance);
// 在當前目標作用域中摧毀指定的作用域中的bean
void destroyScopedBean(String beanName);
// 摧毀在工廠中的所有單例bean
void destroySingletons();

上面的大部分方法都是獲取或者設置一些配置的信息,以便協同來完成BeanFactory的配置。

ConfigurableListableBeanFactory

ConfigurableListableBeanFactory介面繼承自ListableBeanFactory, AutowireCapableBeanFactory, ConfigurableBeanFactory。大多數具有列出能力的bean工廠都應該實現此介面。此了這些介面的能力之外,該介面還提供了分析、修改bean的定義和單例的預先實例化的機制。這個介面不應該用於一般的客戶端代碼中,應該僅僅提供給內部框架使用。下麵是這個介面的方法:

// 忽略用於自動裝配的依賴的類型
void ignoreDependencyType(Class<?> type);
// 忽略用於自動裝配的依賴的介面
void ignoreDependencyInterface(Class<?> ifc);
// 給特定的依賴類型註冊自動裝配的值
void registerResolvableDependency(Class<?> dependencyType, @Nullable Object autowiredValue);
// 指定的bean是否為自動裝配的候選者
boolean isAutowireCandidate(String beanName, DependencyDescriptor descriptor)
    throws NoSuchBeanDefinitionException;
// 獲取bean的定義
BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 獲取這個工廠中的所有bean的名字的迭代器
Iterator<String> getBeanNamesIterator();
// 清除元數據緩存
void clearMetadataCache();
// 凍結所有bean的定義
void freezeConfiguration();
// 工廠中bean的定義是否凍結了
boolean isConfigurationFrozen();
// 對非懶載入的單例進行預先初始化
void preInstantiateSingletons() throws BeansException;

AbstractBeanFactory

AbstractBeanFactory繼承自FactoryBeanRegistrySupport,實現了ConfigurableBeanFactory介面。AbstractBeanFactoryBeanFactory的抽象基礎類實現,提供了完整的ConfigurableBeanFactory的能力。在這裡不討論該抽象類的實現細節,只要知道這個類是乾什麼的就行了,會面會有更加詳細的章節來討論。

  • 單例緩存
  • 別名的管理
  • FactoryBean的處理
  • 用於子bean定義的bean的合併
  • bean的摧毀介面
  • 自定義的摧毀方法
  • BeanFactory的繼承管理

子類需要實現的模板方法如下:

// 是否包含給定名字的bean的定義
protected abstract boolean containsBeanDefinition(String beanName);
// 根據bean的名字來獲取bean的定義,子類通常要實現緩存
protected abstract BeanDefinition getBeanDefinition(String beanName) throws BeansException;
// 為給定的已經合併了的bean的定義創建bean的實例
protected abstract Object createBean(String beanName, RootBeanDefinition mbd, @Nullable Object[] args)
    throws BeanCreationException;

AbstractAutowireCapableBeanFactory

AbstractAutowireCapableBeanFactory繼承自AbstractBeanFactory,實現了AutowireCapableBeanFactory介面。該抽象了實現了預設的bean的創建。

  • 提供了bean的創建、屬性填充、裝配和初始化
  • 處理運行時bean的引用,解析管理的集合、調用初始化方法等
  • 支持構造器自動裝配,根據類型來對屬性進行裝配,根據名字來對屬性進行裝配

子類需要自行實現的模板方法如下:

// 用於根據類型來進行自動裝配
Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName,
                         @Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException;

DefaultListableBeanFactory

DefaultListableBeanFactory繼承自AbstractAutowireCapableBeanFactory,實現了ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable介面。這個類是一個非常完全的BeanFactory,基於bean的定義元數據,通過後置處理器來提供可擴展性。

XmlBeanFactory

XmlBeanFactory繼承自DefaultListableBeanFactory,用來從XML文檔中讀取bean的定義的一個非常方便的類。最底層是委派給XmlBeanDefinitionReader,實際上等價於帶有XmlBeanDefinitionReaderDefaultListableBeanFactory

思維導圖


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在 jQuery 中可以使用 hide() 和 show() 方法來隱藏和顯示 HTML 元素 ...
  • 示例代碼托管在: "http://www.github.com/dashnowords/blogs" 博客園地址: "《大史住在大前端》原創博文目錄" 華為雲社區地址: "【你要的前端打怪升級指南】" [TOC] 一. 需求描述 前端工程出包後實現簡易的自動化部署。 二. 預備知識 網站的建設可以使 ...
  • 舉個慄子 問題描述 上班的日子,上午狀態好,中午想睡覺,下午漸恢復,加班苦煎熬。根據時間的不同體現不同的工作狀態。 簡單實現 Work / 抽象狀態 Created by callmeDevil on 2019/8/3. / public abstract class State { public ...
  • 在RocketMQ中,使用BrokerStartup作為啟動類,相較於NameServer的啟動,Broker作為RocketMQ的核心可複雜得多 【RocketMQ中NameServer的啟動源碼分析】 主函數作為其啟動的入口: 首先通過createBrokerController方法生成Brok ...
  • Bean 的生命周期指的是 Bean 的創建、初始化、銷毀的過程。Spring 提供了一些方法,可以讓開發自定義實現在生命周期過程中執行一些額外操作。 1、在註解 @Bean 中指定初始化和銷毀時執行的方法名。 2、實現初始化和銷毀介面 InitializingBean、DisposableBean ...
  • 1.定義: 在運行狀態中對於任意一個類都能夠知道這個類的所有屬性和方法,對於任意一個對象,都能夠調用它的任意方法和屬性;這種動態獲取信息以及動態調用對象方法的功能稱為java語言的反射機制。 2.應用場景: 編碼階段不知道需要實例化的類名是哪個,需要在runtime從配置文件中載入 在runtime ...
  • 迭代器 迭代器可以用來遍歷字元串、列表、元組、集合、字典。 可以使用next()獲取下一個元素: 錯誤、異常處理 except語句 ecxcept語句用來捕獲、處理錯誤、異常。 as e as是關鍵字,e是e是捕獲的異常實例(對象),可以自己隨便取名。 如果異常處理中用不到捕獲的異常對象,可以不要a ...
  • Java 學習 day01 java的三大技術架構 Javase:java標準版,該體系的知識點主要是學習java基礎的知識點, 主要用於桌面應用軟體的開發。比如計算器,QQ軟體等。==市場上幾乎沒有人使用java去開發桌面應用程式,因為java在創立的時候定位該門語言是面向互聯網的一門語言。Jav ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...