Java之SSM中bean相關知識匯總案例講解
bean 的生命周期
對象創(chuàng)建
- 實例化Bean對象,默認選擇無參構造方法,如果只有一個有參構造那么調用有參構造,如果只有多個有參構造那么報錯,除非其中一個有參構造添加了@AutoWired注解;
- 設置Bean的屬性;
- 依賴注入以及判斷是否實現(xiàn)了Aware相關接口(BeanNameAware, BeanFactoryAware, ApplicationContextAware)
- 如果這個 Bean 關聯(lián)了 BeanPostProcessor 接口,將會調用BeanPostProcessor.postProcessorBeforeInitialization()
- 判斷是否實現(xiàn)了InitalizingBean接口,實現(xiàn)了就執(zhí)行 InitalizingBean.afterPropertiesSet() 方法
- 如果Bean在配置文件中的定義包含init-method屬性(或者添加了@PostConstruct注解),執(zhí)行指定的方法;
- 執(zhí)行方法BeanPostProcessor.postProcessorAfterInitialization():例如判斷有沒有實現(xiàn)AOP
- 創(chuàng)建對象完畢;
對象銷毀
- 當要銷毀Bean的時候,如果Bean實現(xiàn)了DisposableBean接口,執(zhí)行destroy()方法。
- 當要銷毀Bean的時候,如果Bean在配置文件中的定義包含destroy-method屬性(或者添加了@PreDestroy注解),執(zhí)行指定的方法
- 銷毀對象完畢
Bean 的作用域
spring 支持 5 種作用域,如下:
request:每一次HTTP請求都會產(chǎn)生一個新的bean,該bean僅在當前HTTP request內有效。
- singleton::單例模式,Spring IoC 容器中只會存在一個共享的 Bean 實例,無論有多少個Bean 引用它,始終指向同一對象。該模式在多線程下是不安全的。Singleton 作用域是Spring 中的默認作用域
- prototype:每次通過 Spring 容器獲取 prototype 定義的 bean 時,容器都將創(chuàng)建一個新的 Bean 實例,每個 Bean 實例都有自己的屬性和狀態(tài),而 singleton 全局只有一個對象。根據(jù)經(jīng)驗,對有狀態(tài)的bean使用prototype作用域,而對無狀態(tài)的bean使用singleton作用域。
- 用于與數(shù)據(jù)庫交互的存儲數(shù)據(jù)的bean等,均是有狀態(tài)的bean。
- 而僅僅用于操作其他資源的bean,如service controller,就是無狀態(tài)的bean。
- 當然,由于spring使用了ThreadLocal進行多線程處理,絕大多數(shù)bean都可以聲明為singleton作用域。這是后話。
- session:在一次 Http Session 中,容器會返回該 Bean 的同一實例。而對不同的 Session 請求則會創(chuàng)建新的實例,該 bean 實例僅在當前 Session 內有效。同 Http 請求相同,每一次session 請求創(chuàng)建新的實例,而不同的實例之間不共享屬性,且實例僅在自己的 session 請求內有效,請求結束,則實例將被銷毀。如用戶購物車
- global-session:全局session作用域,僅僅在基于Portlet的Web應用中才有意義,Spring5中已經(jīng)沒有了。Portlet是能夠生成語義代碼(例如HTML)片段的小型Java Web插件。它們基于Portlet容器,可以像Servlet一樣處理HTTP請求。但是與Servlet不同,每個Portlet都有不同的會話。
bean的循環(huán)依賴
什么是循環(huán)依賴
一個AService里面引用了BService的一個對象,BService里面又引用了AService的一個對象。
那么在構造AService的bean的時候,會填充屬性以及注入依賴,那么就需要注入BService的bean,spring發(fā)現(xiàn)BService的bean還沒有創(chuàng)建,又會去構造BService的bean。同理BService的bean又需要AService的bean,這時候因為AService的bean還沒有構建好,所以他也會去創(chuàng)建AService的bean。一直循環(huán)
怎么解決:使用二級緩存
- singletonObjects:第一級緩存,里面放置的是實例化好的單例對象;這個是一直存在的
- earlySingletonObjects:第二級緩存,里面存放的是提前曝光的單例對象;就是下面圖中的那個緩存
有什么問題
如果上述的bean不存在AOP,那么是沒有什么問題的,但是如果存在AOP的話,那么構造出來的bean對象就不是原始對象了,而是AOP生成的代理對象。如果還是使用二級緩存的話,那么B從緩存取的是A的原始對象而不是構造好的A的bean對象
怎么解決
添加一層緩存,singletonFactories:第三級緩存,里面存放的是要被實例化的對象的對象工廠。
主要代碼如下:
protected Object getSingleton(String beanName, boolean allowEarlyReference) { Object singletonObject = this.singletonObjects.get(beanName); if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) { synchronized (this.singletonObjects) { singletonObject = this.earlySingletonObjects.get(beanName); if (singletonObject == null && allowEarlyReference) { ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName); if (singletonFactory != null) { singletonObject = singletonFactory.getObject(); this.earlySingletonObjects.put(beanName, singletonObject); this.singletonFactories.remove(beanName); } } } } return (singletonObject != NULL_OBJECT ? singletonObject : null); }
分析getSingleton()的整個過程,Spring首先從一級緩存singletonObjects中獲取。如果獲取不到,并且對象正在創(chuàng)建中,就再從二級緩存earlySingletonObjects中獲取。如果還是獲取不到且允許singletonFactories通過getObject()獲取,就從三級緩存singletonFactory.getObject()(三級緩存)獲取,如果獲取到了則從singletonFactories中移除,并放入earlySingletonObjects中。其實也就是從三級緩存移動到了二級緩存。
總結一下流程:
- A在第一步實例化對象之后將自己提前曝光到singletonFactories中
- 在填充屬性時發(fā)現(xiàn)自己依賴對象B,此時就嘗試去get(B),發(fā)現(xiàn)B還沒有被create,所以走create流程
- B在填充屬性的時候發(fā)現(xiàn)自己依賴了對象A,于是嘗試get(A),嘗試一級緩存singletonObjects(肯定沒有,因為A的bean還沒有構造完),嘗試二級緩存earlySingletonObjects(也沒有),嘗試三級緩存singletonFactories,由于A通過ObjectFactory將自己提前曝光了,所以B能夠通過ObjectFactory.getObject拿到A對象
- 通過提前引用,直接創(chuàng)建出A的動態(tài)代理對象也就是實例化好的A的bean放到第二個緩存中,這樣B的bean就直接實例化完成進入一級緩存。
- 此時回調到A中,A填充屬性這一步完成了繼續(xù)往下執(zhí)行,因為bean是單例的,所以A不會又去調用動態(tài)代理再創(chuàng)建一個bean,而是直接從第二個緩存里拿出實例化好的那個bean出來直接用放進一級緩存中。
到此這篇關于Java之SSM中bean相關知識匯總案例講解的文章就介紹到這了,更多相關Java之SSM中bean相關知識內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Springboot整合logback多節(jié)點日志文件加端口號區(qū)分的操作方法
這篇文章主要介紹了Springboot整合logback多節(jié)點日志文件加端口號區(qū)分的操作方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-09-09Spring配置多個數(shù)據(jù)源并實現(xiàn)數(shù)據(jù)源的動態(tài)切換功能
這篇文章主要介紹了Spring配置多個數(shù)據(jù)源并實現(xiàn)數(shù)據(jù)源的動態(tài)切換功能,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-01-01Spring Boot如何優(yōu)雅的使用多線程實例詳解
這篇文章主要給大家介紹了關于Spring Boot如何優(yōu)雅的使用多線程的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用Spring Boot具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2020-05-05SpringBoot整合RabbitMQ處理死信隊列和延遲隊列
這篇文章將通過示例為大家詳細介紹SpringBoot整合RabbitMQ時如何處理死信隊列和延遲隊列,文中的示例代碼講解詳細,需要的可以參考一下2022-05-05