Spring的同一個(gè)服務(wù)會加載多次的問題分析及解決方法
問題現(xiàn)象
最近在本地調(diào)試公司的一個(gè)Web項(xiàng)目時(shí),無意中發(fā)現(xiàn)日志中出現(xiàn)了兩次同一個(gè)服務(wù)的init記錄,項(xiàng)目都是基于Spring來搭建的,按理說服務(wù)都是單例的,應(yīng)該只有一次服務(wù)加載日志才對,本著對工作認(rèn)真負(fù)責(zé)(閑來無事)的態(tài)度,必然要一探究竟。
問題分析
為什么同一個(gè) Bean 會被容器初始化兩次?
首先,我們先來梳理一下 Web 容器中如何加載 Bean:
在 Web 容器中,ContextLoaderListener
和 DispatchServlet
都會在容器啟動的時(shí)候加載
Bean,區(qū)別在于 DispatchServlet
一般會加載 MVC 相關(guān)的 Bean,ContextLoaderListener
會加載 Spring 相關(guān)的 Bean,二者會分別生成一個(gè)WebApplicationContext
。
根據(jù) web.xml 的加載順序,listener 會先于 Servlet 加載,當(dāng)獲取 Bean 時(shí),會優(yōu)先從DispatchServlet
生成的 WebApplicationContext
中查找,如果找不到再從ContextLoaderListener
生成的 WebApplicationContext
中查找。
那么如果這兩個(gè)加載了同樣的Bean,到底該用誰的呢?
如果二者的配置文件中定義了相同的 Bean,則實(shí)際使用中只會用到 DispatchServlet
中的
Bean,ContextLoaderListener
中的 Bean 無法調(diào)用,造成內(nèi)存泄漏。
接下來我們看一下項(xiàng)目中的 web.xml 配置,如下圖所示,ContextLoaderListener
和DispatchServlet
加載了相同的配置 spring.xml,所以會出現(xiàn)兩次 Bean 的初始化現(xiàn)象。
解決方案
經(jīng)過上面的分析,我們知道了,之所以同一個(gè)Bean會被加載兩次,是由于我們在DispatchServlet
和ContextLoaderListener
都定義了這個(gè)Bean。
因此,我們要做的就是讓ContextLoaderListener
和DispatcherServlet
分別加載不同的Bean:
新增applicationContext.xml,其中聲明ContextLoaderListener要加載的Bean:
修改spring.xml中的包掃描范圍,讓DispatcherServlet只加載mvc相關(guān)的Bean:
啟動服務(wù),查看初始化信息,Service只被初始化了一次:
到此這篇關(guān)于Spring的同一個(gè)服務(wù)為什么會加載多次?的文章就介紹到這了,更多相關(guān)Spring的同一個(gè)服務(wù)為什么會加載多次?內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java:程序包org.bouncycastle.jce.provider不存在問題及解決
這篇文章主要介紹了java:程序包org.bouncycastle.jce.provider不存在問題及解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05SpringBoot項(xiàng)目中java -jar xxx.jar沒有主清單屬性的解決方法
這篇文章主要給大家介紹了SpringBoot項(xiàng)目中java -jar xxx.jar沒有主清單的解決方法,文中通過代碼示例給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-01-01Spring自動裝配Bean實(shí)現(xiàn)過程詳解
這篇文章主要介紹了Spring自動裝配Bean實(shí)現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02Spring Boot+Nginx實(shí)現(xiàn)大文件下載功能
相信很多小伙伴,在日常開放中都會遇到大文件下載的情況,大文件下載方式也有很多,比如非常流行的分片下載、斷點(diǎn)下載;當(dāng)然也可以結(jié)合Nginx來實(shí)現(xiàn)大文件下載,在中小項(xiàng)目非常適合使用,這篇文章主要介紹了Spring Boot結(jié)合Nginx實(shí)現(xiàn)大文件下載,需要的朋友可以參考下2024-05-05java web服務(wù)器實(shí)現(xiàn)跨域訪問
這篇文章主要為大家詳細(xì)介紹了java web服務(wù)器實(shí)現(xiàn)跨域訪問,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-08-08SpringBoot使用CXF集成WebService的方法
這篇文章主要介紹了SpringBoot使用CXF集成WebService的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08