您好,登錄后才能下訂單哦!
這篇文章主要講解了“Spring服務關閉時對象銷毀怎么實現”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Spring服務關閉時對象銷毀怎么實現”吧!
spring提供了兩種方式用于實現對象銷毀時去執行操作
1.實現DisposableBean接口的destroy
2.在bean類的方法上增加@PreDestroy方法,那么這個方法會在DisposableBean.destory方法前觸發
3.實現SmartLifecycle接口的stop方法
package com.wyf.service; import org.springframework.beans.factory.DisposableBean; import org.springframework.context.Lifecycle; import org.springframework.context.SmartLifecycle; import org.springframework.context.annotation.ComponentScan; import org.springframework.stereotype.Component; import javax.annotation.PreDestroy; @Component public class UserService implements DisposableBean, SmartLifecycle { boolean isRunning = false; @Override public void destroy() throws Exception { System.out.println(this.getClass().getSimpleName()+" is destroying....."); } @PreDestroy public void preDestory(){ System.out.println(this.getClass().getSimpleName()+" is pre destory...."); } @Override public void start() { System.out.println(this.getClass().getSimpleName()+" is start.."); isRunning=true; } @Override public void stop() { System.out.println(this.getClass().getSimpleName()+" is stop..."); isRunning=false; } @Override public boolean isRunning() { return isRunning; } }
那么這個時候我們去啟動一個spring容器
package com.wyf; import org.springframework.context.support.ClassPathXmlApplicationContext; public class Application { public static void main(String[] args) { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml"); } }
這個時候其實銷毀方法是不會執行的,我們可以通過,調用close方法觸發或者調用registerShutdownHook注冊一個鉤子來在容器關閉時觸發銷毀方法
package com.wyf; import org.springframework.context.support.ClassPathXmlApplicationContext; public class Application { public static void main(String[] args) { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml"); //添加一個關閉鉤子,用于觸發對象銷毀操作 context.registerShutdownHook(); context.close(); } }
實際上我們去查看源碼會發現本質上這兩種方式都是去調用了同一個方法org.springframework.context.support.AbstractApplicationContext#doClose
@Override public void registerShutdownHook() { if (this.shutdownHook == null) { // No shutdown hook registered yet. this.shutdownHook = new Thread(SHUTDOWN_HOOK_THREAD_NAME) { @Override public void run() { synchronized (startupShutdownMonitor) { doClose(); } } }; Runtime.getRuntime().addShutdownHook(this.shutdownHook); } }
registerShutdownHook
方法其實是創建了一個jvm shutdownhook(關閉鉤子),這個鉤子本質上是一個線程,他會在jvm關閉的時候啟動并執行線程實現的方法。而spring的關閉鉤子實現則是執行了org.springframework.context.support.AbstractApplicationContext#doClose
這個方法去執行一些spring的銷毀方法
@Override public void close() { synchronized (this.startupShutdownMonitor) { doClose(); // If we registered a JVM shutdown hook, we don't need it anymore now: // We've already explicitly closed the context. if (this.shutdownHook != null) { try { Runtime.getRuntime().removeShutdownHook(this.shutdownHook); } catch (IllegalStateException ex) { // ignore - VM is already shutting down } } } }
而close方法則是執行直接執行了doClose
方法,并且在執行之后會判斷是否注冊了關閉鉤子,如果注冊了則注銷掉這個鉤子,因為已經執行過doClose了,不應該再執行一次
@Override public void close() { synchronized (this.startupShutdownMonitor) { doClose(); // If we registered a JVM shutdown hook, we don't need it anymore now: // We've already explicitly closed the context. if (this.shutdownHook != null) { try { Runtime.getRuntime().removeShutdownHook(this.shutdownHook); } catch (IllegalStateException ex) { // ignore - VM is already shutting down } } } }
doClose方法源碼分析
@SuppressWarnings("deprecation") protected void doClose() { // Check whether an actual close attempt is necessary... //判斷是否有必要執行關閉操作 //如果容器正在執行中,并且以CAS的方式設置關閉標識成功,則執行后續關閉操作,當然這個標識僅僅是標識,并沒有真正修改容器的狀態 if (this.active.get() && this.closed.compareAndSet(false, true)) { if (logger.isDebugEnabled()) { logger.debug("Closing " + this); } if (!NativeDetector.inNativeImage()) { LiveBeansView.unregisterApplicationContext(this); } try { // Publish shutdown event. //發布容器關閉事件,通知所有監聽器 publishEvent(new ContextClosedEvent(this)); } catch (Throwable ex) { logger.warn("Exception thrown from ApplicationListener handling ContextClosedEvent", ex); } // Stop all Lifecycle beans, to avoid delays during individual destruction. //如果存在bean實現的Lifecycle接口,則執行onClose(),lifecycleProcessor會對所有Lifecycle進行分組然后分批執行stop方法 if (this.lifecycleProcessor != null) { try { this.lifecycleProcessor.onClose(); } catch (Throwable ex) { logger.warn("Exception thrown from LifecycleProcessor on context close", ex); } } // Destroy all cached singletons in the context's BeanFactory. //銷毀所有緩存的單例bean destroyBeans(); // Close the state of this context itself. //關閉bean工廠 closeBeanFactory(); // Let subclasses do some final clean-up if they wish... //為子類預留的方法允許子類去自定義一些銷毀操作 onClose(); // Reset local application listeners to pre-refresh state. //將本地應用程序偵聽器重置為預刷新狀態。 if (this.earlyApplicationListeners != null) { this.applicationListeners.clear(); this.applicationListeners.addAll(this.earlyApplicationListeners); } // Switch to inactive. //設置上下文到狀態為關閉狀態 this.active.set(false); } }
首先判斷當前容器是否正在運行中,然后嘗試通過CAS的方式設置關閉標識為true,這相當于一個鎖,避免其他線程再去執行關閉操作。
發布容器關閉事件,通知所有監聽器,監聽器收到事件后執行其實現的監聽方法
如果存在bean實現的Lifecycle接口,并且正在運行中,則執行Lifecycle.stop()方法,需要注意的是如果是實現Lifecycle,那么start方法需要使用context.start()去顯示調用才會執行,而實現SmartLifecycle則會自動執行,而stop方法是否執行依賴于isRunning()方法的返回,如果為true那么無論是用哪一種Lifecycle實現,則都會執行stop,當然,你也可以實現isRunning方法讓他默認返回true,那么你也就無需去關注start了。
銷毀所有的單例bean,這里會去執行實現了org.springframework.beans.factory.DisposableBean#destroy
方法的bean的destroy方法,以及其帶有@PreDestroy注解的方法。
關閉Bean工廠,這一步很簡單,就是設置當前上下文持有的bean工廠引用為null即可
執行onClose()方法,這里是為子類預留的擴展,不同的ApplicationContext有不同的實現方式,但是本文主講的不是這個就不談了
將本地應用程序偵聽器重置為預刷新狀態。
將ApplicationContext的狀態設置為關閉狀態,容器正式關閉完成。
tips:其實Lifecycle不算是bean銷毀時的操作,而是bean銷毀前操作,這個是bean生命周期管理實現的接口,相當于spring除了自己去對bean的生命周期管理之外,還允許你通過這個接口來在bean的不同生命周期階段去執行各種邏輯,我個人理解和另外兩種方法的本質上是差不多的,只是誰先執行誰后執行的問題,Lifecycle只不過是把這些能力集成在一個接口里面方便管理和使用。
感謝各位的閱讀,以上就是“Spring服務關閉時對象銷毀怎么實現”的內容了,經過本文的學習后,相信大家對Spring服務關閉時對象銷毀怎么實現這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。