淺嘗Spring註解開發_簡單理解BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor、ApplicationListener

来源:https://www.cnblogs.com/wei-ran/archive/2022/05/04/16221760.html
-Advertisement-
Play Games

淺嘗Spring註解開發_簡單理解BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor、ApplicationListener 淺嘗Spring註解開發,基於Spring 4.3.12 分析BeanFactoryPostProces ...


淺嘗Spring註解開發_簡單理解BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor、ApplicationListener

淺嘗Spring註解開發,基於Spring 4.3.12
分析BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor、ApplicationListener

淺嘗Spring註解開發_自定義註冊組件、屬性賦值、自動裝配
淺嘗Spring註解開發_Bean生命周期及執行過程
淺嘗Spring註解開發_AOP原理及完整過程分析(源碼)
淺嘗Spring註解開發_聲明式事務及原理
淺嘗Spring註解開發_簡單理解BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor、ApplicationListener

BeanFactoryPostProcessor

  • BeanPostProcessor:bean後置處理器,bean創建對象初始化前後進行攔截工作的
  • BeanFactoryPostProcessor:beanFactory的後置處理器
    • BeanFactory標準初始化之後調用,來定製和修改BeanFactory的內容
    • 所有的bean定義已經保存載入到beanFactory,但是bean的實例還未創建

原理

  1. ioc容器創建對象
  2. refresh()->invokeBeanFactoryPostProcessors(beanFactory)
    • 如何找到所有的BeanFactoryPostProcessor並執行他們的方法?
      1. 直接在BeanFactory中找到所有類型是BeanFactoryPostProcessor的組件,
      2. 按照Ordered介面排序
      3. 依次執行它們的方法。在初始化創建其他組件前面執行
@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

	@Override
	public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
		System.out.println("MyBeanFactoryPostProcessor...postProcessBeanFactory...");
		int count = beanFactory.getBeanDefinitionCount();
		String[] names = beanFactory.getBeanDefinitionNames();
		System.out.println("當前BeanFactory中有"+count+" 個Bean");
		System.out.println(Arrays.asList(names));
	}
}

輸出

MyBeanFactoryPostProcessor...postProcessBeanFactory...
當前BeanFactory中有9 個Bean
//[輸出所有BeanDefinitionNames...]
//創建註入容器中的一個Bean
blue...constructor

BeanDefinitionRegistryPostProcessor

  • BeanDefinitionRegistryPostProcessor
    • 繼承自BeanFactoryPostProcessorBeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
    • postProcessBeanDefinitionRegistry()在所有bean定義信息將要被載入,bean實例還未創建
  • 優先於BeanFactoryPostProcessor執行
  • 利用BeanDefinitionRegistryPostProcessor給容器中再額外添加一些組件

原理

  1. ioc創建對象
  2. refresh()->invokeBeanFactoryPostProcessors(beanFactory)
  3. 從容器中獲取到所有的BeanDefinitionRegistryPostProcessor組件
    1. 同樣實現了Ordered介面排序,依次觸發所有的postProcessBeanDefinitionRegistry()方法
    2. 再來觸發BeanFactoryPostProcessor方法postProcessBeanFactory()
@Component
public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor{

	@Override
	public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
		// TODO Auto-generated method stub
		System.out.println("MyBeanDefinitionRegistryPostProcessor...bean的數量:"+beanFactory.getBeanDefinitionCount());
	}

	//BeanDefinitionRegistry Bean定義信息的保存中心,以後BeanFactory就是按照BeanDefinitionRegistry裡面保存的每一個bean定義信息創建bean實例;
	@Override
	public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
		// TODO Auto-generated method stub
		System.out.println("postProcessBeanDefinitionRegistry...bean的數量:"+registry.getBeanDefinitionCount());
        
        //再註冊一個Bean,兩種不同方法
		//RootBeanDefinition beanDefinition = new RootBeanDefinition(Blue.class);
		AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Blue.class).getBeanDefinition();
		registry.registerBeanDefinition("hello", beanDefinition);
	}
}

輸出

//下麵兩個都是MyBeanDefinitionRegistryPostProcessor方法,由於在方法內又手動註冊了一個,所以是11個
postProcessBeanDefinitionRegistry...bean的數量:10
MyBeanDefinitionRegistryPostProcessor...bean的數:11
//下麵一個是BeanFactoryPostProcessor方法
MyBeanFactoryPostProcessor...postProcessBeanFactory...
當前BeanFactory中有11 個Bean
//輸出所有BeanDefinitionNames...
[org.springframework...]
//兩個Bean
blue...constructor
blue...constructor

ApplicationListener

  • ApplicationListener:監聽容器中發佈的事件。事件驅動模型開發

  • public interface ApplicationListener<E extends ApplicationEvent>:監聽 ApplicationEvent 及其下麵的子事件

步驟

  1. 寫一個監聽器(ApplicationListener實現類)來監聽某個事件(ApplicationEvent及其子類),或者使用@EventListener註解標註在監聽方法上
  2. 把監聽器加入到容器
  3. 只要容器中有相關事件的發佈,我們就能監聽到這個事件
    • ContextRefreshedEvent:容器刷新完成(所有bean都完全創建)會發佈這個事件
    • ContextClosedEvent:關閉容器會發佈這個事件
  4. 發佈一個事件
    • applicationContext.publishEvent()

自定義監聽器

@Component
public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {

	//當容器中發佈此事件以後,方法觸發
	@Override
	public void onApplicationEvent(ApplicationEvent event) {
		// TODO Auto-generated method stub
		System.out.println("收到事件:"+event);
	}
}
@Service
public class UserService {
	
	@EventListener(classes={ApplicationEvent.class})
	public void listen(ApplicationEvent event){
		System.out.println("UserService..監聽到的事件:"+event);
	}

}

發佈事件

public class IOCTest_Ext {
	
	@Test
	public void test01(){
		AnnotationConfigApplicationContext applicationContext  = new AnnotationConfigApplicationContext(ExtConfig.class);
		
		//發佈事件;
		applicationContext.publishEvent(new ApplicationEvent(new String("我發佈的事件")) {
		});
		
		applicationContext.close();
	}

}

輸出

//註入Bean
blue...constructor
//監聽容器刷新事件
UserService..監聽到的事件:org.springframework.context.event.ContextRefreshedEvent[]
收到事件:org.springframework.context.event.ContextRefreshedEvent[]
UserService..監聽到的事件:com.atguigu.test.IOCTest_Ext$1[source=我發佈的事件]
//監聽自定義事件
UserService..監聽到的事件:org.springframework.context.event.ContextClosedEvent[]
收到事件:com.atguigu.test.IOCTest_Ext$1[source=我發佈的事件]
//監聽容器關閉事件
UserService..監聽到的事件:org.springframework.context.event.ContextClosedEvent
收到事件:org.springframework.context.event.ContextClosedEvent[]

原理

  1. 發佈ContextRefreshedEvent事件為例:

    1. 容器創建對象:refresh()

    2. finishRefresh(),容器刷新完成

      事件發佈流程:

      1. 獲取事件的多播器(派發器)getApplicationEventMulticaster()

      2. multicastEvent派發事件

      3. 獲取到所有的ApplicationListener

        for (final ApplicationListener<?> listener : getApplicationListeners(event, type))

        1. 如果有Executor,可以支持使用Executor進行非同步派發:Executor executor = getTaskExecutor()
        2. 否則,同步的方式直接執行listener方法:invokeListener(listener, event)
        3. 拿到listener回調onApplicationEvent方法
  2. 發佈自定義事件

  3. 容器關閉發佈ContextClosedEvent事件

事件多播器(派發器)

  1. 容器創建對象:refresh()
  2. initApplicationEventMulticaster():初始化ApplicationEventMulticaster
    1. 先去容器中找有沒有id="applicationEventMulticaster"的組件
    2. 如果沒有就創建一個:this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory),並且加入到容器中,我們就可以在其他組件要派發事件,自動註入這個applicationEventMulticaster

容器中有哪些監聽器

  1. 容器創建對象:refresh()
  2. registerListeners()
    1. 從容器中拿到所有的監聽器,把他們註冊到applicationEventMulticasterString[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false)
    2. listener註冊到ApplicationEventMulticastergetApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName)

SmartInitializingSingleton原理

@EventListener使用EventListenerMethodProcessor處理器來解析方法上的@EventListenerEventListenerMethodProcessor實現了SmartInitializingSingleton

  1. ioc容器創建對象並refresh()

  2. finishBeanFactoryInitialization(beanFactory):初始化剩下的單實例bean

    1. 先創建所有的單實例bean,getBean()

    2. 獲取所有創建好的單實例bean,判斷是否是SmartInitializingSingleton類型的

      如果是就調用afterSingletonsInstantiated()


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

-Advertisement-
Play Games
更多相關文章
  • 本節旨在介紹對於初學者如何學習 Linux 的建議。如果你已經確定對 Linux 產生了興趣,那麼接下來我們介紹一下學習 Linux 的方法。 如何去學習 學習大多類似庖丁解牛,對事物的認識一般都是由淺入深、由表及裡的過程,循序才能漸進。學習 Linux 同樣要有一定的順序和方法,當然這也是你學習本 ...
  • 一、概述 Flink核心是一個流式的數據流執行引擎,並且能夠基於同一個Flink運行時,提供支持流處理和批處理兩種類型應用。其針對數據流的分散式計算提供了數據分佈,數據通信及容錯機制等功能。基於流執行引擎,Flink提供了跟多高抽象層的API便於用戶編寫分散式任務,下麵稍微介紹一下Flink的幾種A ...
  • Hive Hive將HiveQL(類sql語言)轉為MapReduce,完成數據的查詢與分析,減少了編寫MapReduce的複雜度。它有以下優點: 學習成本低:熟悉sql就能使用 良好的數據分析:底層基於MapReduce實現 同樣存在一些缺點: HiveDL表達能力有限 效率不高 Hive調優比較 ...
  • CentOS7.6搭建RAC 1.系統環境配置 1.1概述 ​ 搭建兩個節點的rac集群,其每個節點均有兩個網卡,public網卡和private網卡。兩個節點的主機名分別為rac1和rac2 1.2 參數設置(RAC1&RAC2) 編輯/etc/hosts文件 vim /etc/hosts 註釋掉 ...
  • 前言 一個開源庫,隨著不斷的迭代優化,難免會遇到一個很痛苦的問題 最初的設計並不是很合理:想添加的很多新功能都受此掣肘 想使得該庫更加的強大和健壯,必須要做一個重構 因為重構涉及到對外暴露的api,所以大家會遇到一個比較煩躁的問題:更新版本後,會大面積報錯 我考慮了很久,到底怎麼幫大家快速遷移呢?最 ...
  • Swift字元串追加 var str = "OC" str.append(" Swfit") print(str) // 輸出結果: OC Swift 輸出結果: Swift獲取字元串長度 let str = String(format: "數字%.2f", 333.333) // 獲取長度 pri ...
  • js中關於原型和原型鏈有 __proto__ 、prototype、constructor 頻頻出現在面試題中,但是記得多了反而容易記混。 這裡簡單總結下每個屬性的使用場景,方便記憶。 對象和函數都有 __proto__,對象的 __proto__指向構造函數的prototype,構造函數的__pr ...
  • wait、notify和notifyAll方法 wait() 方法會使該鎖資源釋放,然後線程進入等待WAITING狀態,進入鎖的waitset中,然後等待其他線程對鎖資源調用notify方法或notifyAll方法進行喚醒,否則就會進入無限等待。喚醒後會繼續執行wait() 後面的代碼。 wait( ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...