主頁 > 後端開發 > 【深入淺出Spring原理及實戰】「原始碼除錯分析」深入原始碼探索Spring底層框架的的refresh方法所出現的問題和例外

【深入淺出Spring原理及實戰】「原始碼除錯分析」深入原始碼探索Spring底層框架的的refresh方法所出現的問題和例外

2023-04-24 07:45:20 後端開發

學習Spring原始碼的建議

  1. 閱讀Spring官方檔案,了解Spring框架的基本概念和使用方法,

  2. 下載Spring原始碼,可以從官網或者GitHub上獲取,

  3. 閱讀Spring原始碼的入口類,了解Spring框架的啟動程序和核心組件的加載順序,

  4. 閱讀Spring原始碼中的注釋和檔案,了解每個類和方法的作用和用法,

  5. 除錯Spring原始碼,可以通過IDEA等工具進行除錯,了解Spring框架的內部實作和運行程序,

  6. 參考Spring原始碼的測驗用例,了解Spring框架的各個組件的使用方法和測驗方法,

  7. 參考Spring原始碼的設計模式和最佳實踐,了解如何設計和實作高質量的Java應用程式,

  8. 參與Spring社區,與其他開發者交流和分享經驗,了解Spring框架的最新動態和發展趨勢,


學習Spring原始碼的好處

  1. 更深入地了解Spring框架的內部實作和運行機制,可以更好地理解和使用Spring框架,

  2. 學習Spring原始碼可以提高自己的編程能力和代碼質量,了解Spring框架的設計模式和最佳實踐,可以應用到自己的專案中,

  3. 學習Spring原始碼可以幫助開發者解決一些復雜的問題和難點,提高自己的解決問題的能力,

  4. 學習Spring原始碼可以幫助開發者更好地理解Java語言和面向物件編程的思想,提高自己的編程水平,

  5. 學習Spring原始碼可以幫助開發者更好地了解Java生態系統和相關技術,如AOP、IOC、MVC等,

  6. 學習Spring原始碼可以幫助開發者更好地了解開源軟體的開發和維護程序,提高自己的開源軟體開發能力,

refresh方法所出現的問題和例外

最近抽空總結一下之前通用的Spring框架所出現的問題和例外情況,當創建屬于自己的ApplicationContext物件的時候,經常會遇到這么幾條例外訊息:

  1. LifecycleProcessor not initialized - call 'refresh' before invoking lifecycle methods via the context: ......

LifecycleProcessor物件沒有初始化,在呼叫context的生命周期方法之前必須呼叫'refresh'方法,

  1. BeanFactory not initialized or already closed - call 'refresh' before accessing beans via the ApplicationContext

BeanFactory物件沒有初始化或已經關閉了,使用ApplicationContext獲取Bean之前必須呼叫'refresh'方法,

  1. ApplicationEventMulticaster not initialized - call 'refresh' before multicasting events via the context: ......

ApplicationEventMulticaster物件沒有初始化,在context廣播事件之前必須呼叫'refresh'方法,

這幾條例外訊息都與refresh方法有關,那拋出這些例外的原因到底是什么,為什么在這么多情況下一定要先呼叫refresh方法(定義在AbstractApplicationContext類中),在此這前我們先看看refresh方法中又干了些什么?

public void refresh() throws BeansException, IllegalStateException {
    synchronized (this.startupShutdownMonitor) {
        //重繪之前的準備作業,包括設定啟動時間,是否激活標識位,初始化屬性源(property source)配置
        prepareRefresh();
        //由子類去重繪BeanFactory(如果還沒創建則創建),并將BeanFactory回傳
        ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
        //準備BeanFactory以供ApplicationContext使用
        prepareBeanFactory(beanFactory);
        try {
            //子類可通過修改此方法來對BeanFactory進行修改
            postProcessBeanFactory(beanFactory);
            //實體化并呼叫所有注冊的BeanFactoryPostProcessor物件
            invokeBeanFactoryPostProcessors(beanFactory);
            //實體化并呼叫所有注冊的BeanPostProcessor物件
            registerBeanPostProcessors(beanFactory);
            //初始化MessageSource
            initMessageSource();
            //初始化事件廣播器
            initApplicationEventMulticaster();
            //子類覆寫此方法在重繪程序欄位外作業
            onRefresh();
            //注冊應用監聽器ApplicationListener
            registerListeners();
            //實體化所有non-lazy-init bean
            finishBeanFactoryInitialization(beanFactory);
            //重繪完成作業,包括初始化LifecycleProcessor,發布重繪完成事件等
            finishRefresh();
        }
        catch (BeansException ex) {
            // Destroy already created singletons to avoid dangling resources.
            destroyBeans();
            // Reset 'active' flag.
            cancelRefresh(ex);
            // Propagate exception to caller.
            throw ex;
        }
    }
}

與此三條例外訊息相關的方法分別為:

finishRefresh

LifecycleProcessor not initialized - call 'refresh' before invoking lifecycle methods via the context:

protected void finishRefresh() {
    // //初始化LifecycleProcessor
    initLifecycleProcessor();
    // Propagate refresh to lifecycle processor first.
    getLifecycleProcessor().onRefresh();
    // Publish the final event.
    publishEvent(new ContextRefreshedEvent(this));
    // Participate in LiveBeansView MBean, if active.
    LiveBeansView.registerApplicationContext(this);
}

如果沒有呼叫finishRefresh方法,則lifecycleProcessor成員為null,

obtainFreshBeanFactory

protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {
    refreshBeanFactory();//重繪BeanFactory,如果beanFactory為null,則創建
    ConfigurableListableBeanFactory beanFactory = getBeanFactory();
    if (logger.isDebugEnabled()) {
        logger.debug("Bean factory for " + getDisplayName() + ": " + beanFactory);
    }
    return beanFactory;
}

refreshBeanFactory()為一抽象方法,真正實作在AbstractRefreshableApplicationContext類中:

@Override
protected final void refreshBeanFactory() throws BeansException {
	//如果beanFactory已經不為null,則銷毀beanFactory中的Bean后自行關閉
    if (hasBeanFactory()) {
        destroyBeans();
        closeBeanFactory();
    }
    try {
        DefaultListableBeanFactory beanFactory = createBeanFactory();//創建beanFactory
        beanFactory.setSerializationId(getId());
        customizeBeanFactory(beanFactory);
        loadBeanDefinitions(beanFactory);
        synchronized (this.beanFactoryMonitor) {
            this.beanFactory = beanFactory;//對beanFactory成員進行賦值
        }
    }
    catch (IOException ex) {
        throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);
    }
}

如果沒有呼叫obtainFreshBeanFactory()方法則beanFactory成員為null,

initApplicationEventMulticaster

protected void initApplicationEventMulticaster() {
    ConfigurableListableBeanFactory beanFactory = getBeanFactory();
    if (beanFactory.containsLocalBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME)) {
        this.applicationEventMulticaster =
                beanFactory.getBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, ApplicationEventMulticaster.class);
        if (logger.isDebugEnabled()) {
            logger.debug("Using ApplicationEventMulticaster [" + this.applicationEventMulticaster + "]");
        }
    }
    else {
        this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);
        beanFactory.registerSingleton(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, this.applicationEventMulticaster);
        if (logger.isDebugEnabled()) {
            logger.debug("Unable to locate ApplicationEventMulticaster with name '" +
                    APPLICATION_EVENT_MULTICASTER_BEAN_NAME +
                    "': using default [" + this.applicationEventMulticaster + "]");
        }
    }
}

而這三個方法呼叫都在refresh()方法中,由上面的分析可知,如果沒有呼叫refresh方法,則背景關系中的lifecycleProcessor,beanFactory,applicationEventMulticaster成員都會為null,至此可以來詳細分析這三條例外訊息的緣由了,

下面是針對上面三條例外訊息的三段測驗代碼,順序相對應:

例外的測驗案例(1)

public static void main(String[] args) {
    ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext();
    applicationContext.setConfigLocation("application-context.xml");
    applicationContext.start();
    applicationContext.close();
}

對于第一條例外訊息,例外堆疊出錯在applicationContext.start();下面是start()方法原始碼:

public void start() {
    getLifecycleProcessor().start();
    publishEvent(new ContextStartedEvent(this));
}

可以看到start()方法中要先獲取lifecycleProcessor物件,而默認構造方法中并沒用呼叫refresh方法,所以lifecycleProcessor為null,故而在getLifecycleProcessor()方法中拋出了此例外訊息,

例外的測驗案例(2)

public static void main(String[] args) {
    ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext();
    applicationContext.setConfigLocation("application-context.xml");
    applicationContext.getBean("xtayfjpk");
    applicationContext.close();
}

第二條例外訊息,例外堆疊出錯在applicationContext.getBean("xtayfjpk"),applicationContext.getBean()方法呼叫的是背景關系中beanFactory的getBean()方法實作的,獲取BeanFactory物件的代碼在其基類ConfigurableListableBeanFactory中的getBeanFactory()方法中:

@Override
public final ConfigurableListableBeanFactory getBeanFactory() {
    synchronized (this.beanFactoryMonitor) {
        if (this.beanFactory == null) {
            throw new IllegalStateException("BeanFactory not initialized or already closed - " +
                    "call 'refresh' before accessing beans via the ApplicationContext");
        }
        return this.beanFactory;
    }
}

由于ClassPathXmlApplicationContext的默認構造方法沒有呼叫refresh()方法,所以beanFactory為null,因此拋出例外,

例外的測驗案例(3)

public static void main(String[] args) {
    GenericApplicationContext parent = new GenericApplicationContext();
    AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();
    context.setParent(parent);
    context.refresh();
    context.start();
    context.close();
}

這其中提到了生命周期方法,其實就是定義在org.springframework.context.Lifecycle介面中的start(), stop(), isRunning()三個方法,如果是剛開始學習Spring的話,創建ClassPathXmlApplicationContext物件時應該是這樣的:ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("application-context.xml"),

這樣直接呼叫start()方法卻又不會出現例外,這是為什么呢?這是因為ClassPathXmlApplicationContext(String configLocation)這個構造方法最終呼叫的是:

public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent) throws BeansException {
    super(parent);
    setConfigLocations(configLocations);
    if (refresh) {//refresh傳遞值為true,這樣就自動呼叫了refresh方法進行了重繪
        refresh();
    }
}

第三條例外訊息,例外堆疊出錯在context.refresh(),但是如果沒有設定父背景關系的話context.setParent(parent),例子代碼是不會出現例外的,這是因為在refresh方法中的finishRefresh()方法呼叫了publishEvent方法:

public void publishEvent(ApplicationEvent event) {
    Assert.notNull(event, "Event must not be null");
    if (logger.isTraceEnabled()) {
        logger.trace("Publishing event in " + getDisplayName() + ": " + event);
    }
    getApplicationEventMulticaster().multicastEvent(event);
    if (this.parent != null) {
        this.parent.publishEvent(event);
    }
}

從上面可以看到:如果父背景關系不為null,則還需要呼叫父容器的pushlishEvent方法,而且在該方法中呼叫了getApplicationEventMulticaster()方法以獲取一個事件廣播器,問題就出現在這里:

private ApplicationEventMulticaster getApplicationEventMulticaster() throws IllegalStateException {
    if (this.applicationEventMulticaster == null) {//如果為null則拋例外
        throw new IllegalStateException("ApplicationEventMulticaster not initialized - " +
                "call 'refresh' before multicasting events via the context: " + this);
    }
    return this.applicationEventMulticaster;
}

而applicationEventMulticaster就是在refresh方法中的initApplicationEventMulticaster方法在實體化的,則于父背景關系沒有呼叫過refresh方法,所以父背景關系的applicationEventMulticaster成員為null,因此拋出例外,

問題總結

綜上所述,其實這三條例外訊息的根本原因只有一個,就是當一個背景關系物件創建后沒有呼叫refresh()方法,在Spring中ApplicationContext實作類有很多,有些實作類在創建的程序中自動呼叫了refresh()方法,而有些又沒有,如果沒有則需要自己手動呼叫refresh()方法,一般說來實作WebApplicationContext介面的實作類以及使用默認構造方法創建背景關系物件時不會自動refresh()方法,其它情況則會自動呼叫,

本文來自博客園,作者:洛神灬殤,轉載請注明原文鏈接:https://www.cnblogs.com/liboware/p/17347682.html,任何足夠先進的科技,都與魔法無異,

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

標籤:其他

上一篇:最短路徑問題

下一篇:返回列表

標籤雲
其他(157908) Python(38094) JavaScript(25383) Java(17988) C(15215) 區塊鏈(8258) C#(7972) AI(7469) 爪哇(7425) MySQL(7137) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4558) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2430) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1959) Web開發(1951) HtmlCss(1921) python-3.x(1918) 弹簧靴(1913) C++(1910) xml(1889) PostgreSQL(1872) .NETCore(1854) 谷歌表格(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原理及實戰】「原始碼除錯分析」深入原始碼探索

    學習Spring原始碼的建議 閱讀Spring官方檔案,了解Spring框架的基本概念和使用方法。 下載Spring原始碼,可以從官網或者GitHub上獲取。 閱讀Spring原始碼的入口類,了解Spring框架的啟動程序和核心組件的加載順序。 閱讀Spring原始碼中的注釋和檔案,了解每個類和方法的作用和 ......

    uj5u.com 2023-04-24 07:45:20 more
  • 最短路徑問題

    平面上有n個點(n<=100),每個點的坐標均在-10000~10000之間,其中的一些點之間有連線。 若有連線,則表示可從一個點到達另一個點,即兩點間有通路,同路的距離為兩點間的直線距離。現在的任務是找出從一點到另一點之間的最短路徑。 小提示: 兩點的距離:如果點$A$坐標為$(x_A,y_A)$ ......

    uj5u.com 2023-04-24 07:45:15 more
  • Django筆記二十九之中間件介紹

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十九之中間件介紹 這一節介紹一下 Django 的中間件。 關于中間件,官方檔案的解釋為:中間件是一個嵌入 Django 系統的 request 和 response 的鉤子框架,是一個能夠全域改變 Django 輸入/輸出的系統。 我 ......

    uj5u.com 2023-04-24 07:45:08 more
  • 15面向物件特性

    面向物件特性 封裝 在程式設計中,封裝(Encapsulation)是對具體物件的一種抽象,即將某些部分隱藏起來,在程式外部看不到,其含義是其他程式無法呼叫。要了解封裝,離不開“私有化”,就是將類或者是函式中的某些屬性限制在某個區域之內,外部無法呼叫。 封裝的作用: 1、保護隱私(把不想別人知道的東 ......

    uj5u.com 2023-04-24 07:45:03 more
  • 【Qt6】QWindow類可以做什么

    原來的水文標題是“用 VS Code 搞 Qt6”,想想還是直接改為“Qt6”,反正這個用不用 VS Code 也能搞。雖然我知道大伙伴們都很討厭 CMake,但畢竟這廝幾乎成了 C++ 的玩家規范了。Qt 也算識大體,支持用 CMake 來構建程式。所以,只要你用的是能寫 C++ 的工具,理論上都 ......

    uj5u.com 2023-04-24 07:44:51 more
  • docker常用命令

    #一、Docker基本概念 ###1.鏡像(Image) Docker 鏡像 是一個特殊的檔案系統,除了提供容器運行時所需的程式、庫、資源、配置等檔案外,還包含了一些為運行時準備的一些配置引數(如匿名卷、環境變數、用戶等)。鏡像 不包含 任何動態資料,其內容在構建之后也不會被改變。 docker的鏡 ......

    uj5u.com 2023-04-24 07:44:44 more
  • springboot~關于md5簽名引發的問題

    事實是這樣的,我有個介面,這個介面不能被篡改,于是想到了比較簡單的md5對url地址引數進行加密,把這個密碼當成是sign,然后服務端收到請求后,使用相同演算法也生成sign,兩個sign相同就正常沒有被篡改過。 問題的出現 介面中的引數包括userId,extUserId,時間,其中extUserI ......

    uj5u.com 2023-04-24 07:44:38 more
  • Java中代碼的執行順序

    結論 注意 只有顯式的加載類 JVM才會加載到記憶體中 先加載父類的靜態代碼塊 然后執行子類靜態代碼塊 當前類存在類靜態變數注意參考型別沒進行賦值操作初始化為null 并不會顯式的加載類又存在靜態代碼塊 會先執行前者進行初始化 再執行靜態代碼塊 在實體化類的時候 執行順序 構造代碼塊-->構造方法存在 ......

    uj5u.com 2023-04-24 07:44:33 more
  • Nginx配置跨域,覆寫后端服務跨域配置

    本篇文章主要介紹了,如何通過Nginx配置跨域,并覆寫后端服務跨域配置。 先看下后端代碼跨域配置: 主要的目標是:不修改后端跨域配置代碼,來實作Nginx跨域指定域名。 @Bean public CorsFilter corsFilter() { final UrlBasedCorsConfigur ......

    uj5u.com 2023-04-24 07:44:29 more
  • Maven的使用

    Maven 1.下載并配置 下載地址:https://maven.apache.org/download.cgi?. 配置環境變數 新建系統變數,變數名為MAVEN_HOME,變數值為 maven 的安裝路徑 編輯名為Path的系統變數,然后點擊新建,輸入 %MAVEN_HOME%\bin 配置完成 ......

    uj5u.com 2023-04-24 07:44:14 more