主頁 > 後端開發 > Spring原始碼:Bean生命周期(終章)

Spring原始碼:Bean生命周期(終章)

2023-05-24 07:56:54 後端開發

前言

本系列前面講解了Spring的bean定義、bean實體化、bean初始化等生命周期,這些步驟使我們能夠了解bean從創建到準備好使用所經歷的程序,但是,除了這些步驟,bean的銷毀也是非常重要的一步,在本系列的最后,我們將深入探討bean的銷毀程序,包括在什么情況下會發生銷毀、銷毀的順序以及如何在bean銷毀之前執行一些清理任務等,通過學習bean的銷毀程序,我們將更全面地了解Spring的bean生命周期,

在Spring中,有多種方式可以銷毀bean,其中一種方式是在應用程式關閉時顯式地呼叫applicationContext.close()方法來關閉容器,這個方法將會銷毀所有還沒有被銷毀的bean,

AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);

applicationContext.close();

實作DisposableBean介面

實作DisposableBean介面是一種銷毀bean的簡單方式,當bean容器關閉時,Spring會呼叫DisposableBean的destroy()方法來銷毀bean,以下是一些示例代碼:

import org.springframework.beans.factory.DisposableBean;
@Component
public class MyBean implements DisposableBean {

    @Override
    public void destroy() throws Exception {
        // 在這里清理資源
    }
}

使用@PreDestroy注解

使用@PreDestroy注解是另一種簡單的方式來銷毀bean,當bean容器關閉時,Spring會呼叫使用@PreDestroy注解的方法來銷毀bean,以下是一些示例代碼:

import javax.annotation.PreDestroy;
@Component
public class MyBean {

    @PreDestroy
    public void cleanUp() throws Exception {
        // 在這里清理資源
    }
}

registerDisposableBeanIfNecessary

registerDisposableBeanIfNecessary()方法是一個非常重要的方法,它是在bean創建后進行處理bean銷毀邏輯的前提,在Spring的AbstractBeanFactory類中,該方法會檢查當前bean是否實作了DisposableBean介面或者@PreDestroy注解,如果是的話,就會將該bean添加到一個DisposableBeanAdapter物件中,該物件會在bean銷毀時被呼叫以執行銷毀任務,這個程序是在bean銷毀之前執行的,以確保正確關閉應用程式,

    protected void registerDisposableBeanIfNecessary(String beanName, Object bean, RootBeanDefinition mbd) {
        AccessControlContext acc = (System.getSecurityManager() != null ? getAccessControlContext() : null);
        if (!mbd.isPrototype() && requiresDestruction(bean, mbd)) {
            if (mbd.isSingleton()) {
                // Register a DisposableBean implementation that performs all destruction
                // work for the given bean: DestructionAwareBeanPostProcessors,
                // DisposableBean interface, custom destroy method.
                registerDisposableBean(beanName, new DisposableBeanAdapter(
                        bean, beanName, mbd, getBeanPostProcessorCache().destructionAware, acc));
            }
            else {
                // A bean with a custom scope...
                Scope scope = this.scopes.get(mbd.getScope());
                if (scope == null) {
                    throw new IllegalStateException("No Scope registered for scope name '" + mbd.getScope() + "'");
                }
                scope.registerDestructionCallback(beanName, new DisposableBeanAdapter(
                        bean, beanName, mbd, getBeanPostProcessorCache().destructionAware, acc));
            }
        }
    }

我大概講下這個方法requiresDestruction

    protected boolean requiresDestruction(Object bean, RootBeanDefinition mbd) {
        return (bean.getClass() != NullBean.class && (DisposableBeanAdapter.hasDestroyMethod(bean, mbd) ||
                (hasDestructionAwareBeanPostProcessors() && DisposableBeanAdapter.hasApplicableProcessors(
                        bean, getBeanPostProcessorCache().destructionAware))));
    }
  1. DisposableBeanAdapter.hasDestroyMethod:校驗是否實作了DisposableBean或者AutoCloseable介面,如果沒有的話,再查看是否bean定義的destroyMethodName屬性是(inferred),如果是的話,那么直接找這個類是否有close方法沒有的話再找shutdown方法
  2. DisposableBeanAdapter.hasApplicableProcessors:是否有@PreDestroy注解

DisposableBeanAdapter

DisposableBeanAdapter物件是一個配接器,用于在銷毀bean時執行必要的處理,它會將DisposableBean介面或@PreDestroy注解的方法轉換為一個回呼方法,以便在bean銷毀時執行,這種配接器模式允許非標準的bean銷毀方法與Spring框架協同作業,

在將DisposableBeanAdapter物件添加到一個DisposableBeanRegistry物件中時,Spring會將該物件添加到一個bean銷毀的注冊表中,當需要銷毀所有bean時,Spring就會從該注冊表中獲取所有需要銷毀的bean,并按照正確的順序執行銷毀任務,這樣就可以確保應用程式的正確關閉,

destroySingleton

當Spring程式關閉時,會呼叫destroyBeans方法,這里我們分析關鍵部分代碼:

    public void destroySingleton(String beanName) {
        // Remove a registered singleton of the given name, if any.
        // 先從單例池中移除掉
        removeSingleton(beanName);

        // Destroy the corresponding DisposableBean instance.
        DisposableBean disposableBean;
        synchronized (this.disposableBeans) {
            disposableBean = (DisposableBean) this.disposableBeans.remove(beanName);
        }
        destroyBean(beanName, disposableBean);
    }
  1. removeSingleton:先從單例池中移除掉
  2. this.disposableBeans.remove:這里回傳的是我們之前呼叫registerDisposableBeanIfNecessary方法添加進去的DisposableBeanAdapter配接器
  3. destroyBean:直接銷毀bean,這里注意一個小點就是如果當前bean被其他bean依賴了,那么先移除銷毀其他Bean,然后就是呼叫配接器的destroy方法

總結

非常感謝您對 Spring 生命周期系列文章的關注和支持,我們在過去一個月中深入了解了 Spring 框架中 Bean 的生成、初始化、后置處理和銷毀等程序,對于理解 Spring 框架的原理和機制非常有幫助,我們總結一下Spring到底做了那些事情將bean從生成到銷毀的全程序:

  1. 專案啟動時,ClassPathBeanDefinitionScanner掃描得到所有BeanDefinition,由于ACM技術所以此時beanclass屬性為String型別的bean的名稱
  2. 獲取合并后的BeanDefinition
  3. beanClass開始真正的被加載替換原有String型別的bean的名稱
  4. 呼叫實體化前處理方法applyBeanPostProcessorsBeforeInstantiation
  5. 通過構造方法創建Bean實體
  6. 后置處理合并后的BeanDefinition,呼叫postProcessMergedBeanDefinition(尋找注入點)
  7. 呼叫實體化后處理方法postProcessAfterInstantiation
  8. 開始進行屬性注入:postProcessProperties
  9. 呼叫初始化前處理方法:applyBeanPostProcessorsBeforeInitialization
  10. 進行初始化:invokeInitMethods,會呼叫指定init方法或者afterPropertiesSet方法
  11. 呼叫初始化后處理方法:applyBeanPostProcessorsAfterInitialization(AOP)
  12. 容器關閉時,走bean的銷毀邏輯,即今天所講

這里面有很多邏輯流程我都在單獨的文章中有細講,比如FactoryBean、PropertyValues等等,由于是總結所以就不全寫出來了,也希望大家可以好好理解Spring原始碼,下一步,我們將會著重講解 Bean 的屬性依賴注入,

公眾號 ps:以上內容,純屬個人見解,有任何問題下方評論!關注博主公眾號,原始碼專題、面試精選、AI最新擴展等你來看!原創撰寫不易,轉載請說明出處!

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/553194.html

標籤:Java

上一篇:java IO流

下一篇:返回列表

標籤雲
其他(159545) Python(38162) JavaScript(25444) Java(18109) C(15231) 區塊鏈(8267) C#(7972) AI(7469) 爪哇(7425) MySQL(7207) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5340) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4575) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2433) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1976) 功能(1967) Web開發(1951) HtmlCss(1941) C++(1922) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1878) .NETCore(1861) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Spring原始碼:Bean生命周期(終章)

    本系列前面講解了Spring的bean定義、bean實體化、bean初始化等生命周期階段。這些步驟使我們能夠了解bean從創建到準備好使用所經歷的程序。但是,除了這些步驟,bean的銷毀也是非常重要的一步。在本系列的最后,我們將深入探討bean的銷毀程序,包括在什么情況下會發生銷毀、銷毀的順序以及如... ......

    uj5u.com 2023-05-24 07:56:54 more
  • java IO流

    # Java IO流 ## 什么是流? 概念:記憶體和存盤設備之間傳輸資料的通道。 資料借助流傳輸。 流分類: - 按照方向:輸入流(將存盤設備中的內容讀入到記憶體中)和輸出流(將記憶體中的內容寫入到存盤設備中) - 按照單位:位元組流(以位元組為單位,可以讀寫所有資料)和字符流(以字符為單位,只能讀取文本數 ......

    uj5u.com 2023-05-24 07:56:38 more
  • 小愛同學呼叫本地jar -巴法云

    本文使用的是巴法云 你也可以使用其他的物聯網平臺 并且 也不一定是小愛 比如小度啊 等等其他的一下應該也是可以實作的
    調到java里面之后 剩下的事情大家就可以想干嘛就干嘛了 ......

    uj5u.com 2023-05-24 07:56:24 more
  • 解決啟動jar包報錯 錯誤 找不到或無法加載主類 jar

    #### 錯誤: 找不到或無法加載主類 jar ##### 問題描述: 在使用springboot框架對專案打包后,手動使用命令java -jar 包名啟動jar包,報錯:錯誤: 找不到或無法加載主類 jar。 網上找了各辦法,都是加maven插件,打成可執行jar包 ``` org.springf ......

    uj5u.com 2023-05-24 07:56:10 more
  • Netty實戰(一)

    [TOC](Nett的概念及體系結構) # 第一章 Java網路編程 最早期的 Java API(java.net)只支持由本地系統套接字庫提供的所謂的阻塞函式,像下面的那樣 ```java //創建一個新的 ServerSocket,用以監聽指定埠上的連接請求 ServerSocket serv ......

    uj5u.com 2023-05-24 07:56:02 more
  • Spring Boot 3.1 正式發布,王炸!!

    ## Spring Boot 3.1 正式發布 大家好,我是R哥。 上一篇:[Spring Boot 3.0 正式發布,王炸!!](https://mp.weixin.qq.com/s/p-rDuyNv68hQvwRBrm5KWA) Spring Boot 3.0 發布半年左右,Spring Boo ......

    uj5u.com 2023-05-24 07:55:54 more
  • springcloud~gateway網關

    有時間,我們在搭建微服務時,總希望拿一個比較單純的,沒有污染其它代碼的專案來從頭開始做,今天我們來建設一個最簡單的,gateway專案,它被注冊到nacos里,路由配置也存到nacos里,動態實作更新配置功能。 # 依賴配置 > 版本:com.alibaba.cloud:spring-cloud-s ......

    uj5u.com 2023-05-24 07:50:39 more
  • 學習筆記-Spring事務

    學習的文章 [小姐姐非要問我:spring編程式事務是啥? (qq.com)](https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648936779&idx=2&sn=a6255c7d436a62af380dfa6b326fd4e7&chk ......

    uj5u.com 2023-05-24 07:45:32 more
  • 【2023.03.20】P4710 「物理」平拋運動

    題目傳送門: >[【洛谷】P4710 [物理]平拋運動](https://www.luogu.com.cn/problem/P4710 "【洛谷】P4710 [物理]平拋運動") ## Step 1:前置芝士 您需要知道并了解以下芝士: 1. 數學: - 三角函式; 2. 物理: - 加速度公式; ......

    uj5u.com 2023-05-24 07:44:42 more
  • 驅動開發:內核實作行程匯編與反匯編

    在筆者上一篇文章`《驅動開發:內核MDL讀寫行程記憶體》`簡單介紹了如何通過MDL映射的方式實作行程讀寫操作,本章將通過如上案例實作遠程行程反匯編功能,此類功能也是ARK工具中最常見的功能之一,通常此類功能的實作分為兩部分,內核部分只負責讀寫位元組集,應用層部分則配合反匯編引擎對位元組集進行解碼,此處我們... ......

    uj5u.com 2023-05-24 07:44:33 more