欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring中SmartLifecycle和Lifecycle的作用和區(qū)別

 更新時(shí)間:2021年03月10日 09:01:16   作者:brucelwl  
這篇文章主要介紹了Spring中SmartLifecycle和Lifecycle的作用和區(qū)別,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì)對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下

本文基于SpringBoot 2.5.0-M2講解Spring中LifecycleSmartLifecycle的作用和區(qū)別,以及如何控制SmartLifecycle的優(yōu)先級(jí)。
并講解SpringBoot中如何通過(guò)SmartLifecycle來(lái)啟動(dòng)/停止web容器.

 SmartLifecycle和Lifecycle作用

都是讓開(kāi)發(fā)者可以在所有的bean都創(chuàng)建完成(getBean) 之后執(zhí)行自己的初始化工作,或者在退出時(shí)執(zhí)行資源銷(xiāo)毀工作。

SmartLifecycle和Lifecycle區(qū)別

 1.SmartLifecycle接口繼承Lifecycle接口,同時(shí)繼承了org.springframework.context.Phased接口用于控制多個(gè)SmartLifecycle實(shí)現(xiàn)之間的優(yōu)先級(jí)。

2.在SpringBoot應(yīng)用中,或在Spring應(yīng)用中沒(méi)有調(diào)用AbstractApplicationContext#start方法,如果一個(gè)Bean只是實(shí)現(xiàn)了Lifecycle接口的情況下:

不會(huì)執(zhí)行Lifecycle接口中的啟動(dòng)方法,包括Lifecycle#isRunning方法也不會(huì)被執(zhí)行。

但是在應(yīng)用 退出時(shí) 會(huì)執(zhí)行Lifecycle#isRunning方法判斷該Lifecycle是否已經(jīng)啟動(dòng),如果返回true則調(diào)用Lifecycle#stop()停止方法。

3. 如果一個(gè)Bean實(shí)現(xiàn)了SmartLifecycle接口,則會(huì)執(zhí)行啟動(dòng)方法。先會(huì)被根據(jù)Phased接口優(yōu)先級(jí)分組,封裝在LifecycleGroup,然后循環(huán)調(diào)用LifecycleGroup#start()方法,SmartLifecycle#isRunning判斷是否已經(jīng)執(zhí)行,返回false表示還未執(zhí)行,則調(diào)用SmartLifecycle#start()執(zhí)行。Phased返回值越小,優(yōu)先級(jí)越高。

4.SmartLifecycle中還有個(gè)isAutoStartup方法,如果返回false,在啟動(dòng)時(shí)也不會(huì)執(zhí)行start方法,默認(rèn)返回true

源碼分析

SmartLifecycleLifecycle都是在org.springframework.context.support.DefaultLifecycleProcessor中被調(diào)用,
DefaultLifecycleProcessor#onRefresh方法在執(zhí)行AbstractApplicationContext#finishRefresh時(shí)會(huì)被調(diào)用,調(diào)用棧如下:

startBeans:142, DefaultLifecycleProcessor (org.springframework.context.support)
onRefresh:123, DefaultLifecycleProcessor (org.springframework.context.support)
finishRefresh:934, AbstractApplicationContext (org.springframework.context.support)
refresh:585, AbstractApplicationContext (org.springframework.context.support)
refresh:144, ServletWebServerApplicationContext (org.springframework.boot.web.servlet.context)
refresh:755, SpringApplication (org.springframework.boot)
refreshContext:426, SpringApplication (org.springframework.boot)
run:326, SpringApplication (org.springframework.boot)
run:1299, SpringApplication (org.springframework.boot)
run:1288, SpringApplication (org.springframework.boot)
main:31, DemoApplication (com.example.demo)

DefaultLifecycleProcessor#onRefresh源碼:

@Override
public void onRefresh() {
	startBeans(true); //autoStartupOnly = true
	this.running = true;
}

DefaultLifecycleProcessor#startBeans源碼如下:
autoStartupOnly 在onRefresh時(shí)傳入的是true,表示只執(zhí)行可以自動(dòng)啟動(dòng)的bean,即為:SmartLifecycle的實(shí)現(xiàn)類(lèi),并且SmartLifecycle#isAutoStartup返回值必須為true。

private void startBeans(boolean autoStartupOnly) {
	Map<String, Lifecycle> lifecycleBeans = getLifecycleBeans();
	Map<Integer, LifecycleGroup> phases = new TreeMap<>();

	lifecycleBeans.forEach((beanName, bean) -> {
		if (!autoStartupOnly || (bean instanceof SmartLifecycle && ((SmartLifecycle) bean).isAutoStartup())) {
			int phase = getPhase(bean);
			phases.computeIfAbsent(phase, p -> 
			 new LifecycleGroup(phase, this.timeoutPerShutdownPhase, lifecycleBeans, autoStartupOnly)
			).add(beanName, bean);
		}
	});
	if (!phases.isEmpty()) {
		phases.values().forEach(LifecycleGroup::start);
	}
}

而Spring AbstractApplicationContext#doClose退出時(shí),無(wú)論是SmartLifecycleLifecycle都會(huì)執(zhí)行isRunning方法,判斷是否已經(jīng)啟動(dòng),返回true表示已經(jīng)啟動(dòng),則執(zhí)行SmartLifecycleLifecyclestop方法。
源碼見(jiàn):org.springframework.context.support.DefaultLifecycleProcessor#doStop方法。

而執(zhí)行AbstractApplicationContext#doClose一般是應(yīng)用進(jìn)程退出,通過(guò)jvm注冊(cè)的鉤子方法,或者應(yīng)用程序編碼調(diào)用。
AbstractApplicationContext#registerShutdownHook源碼

@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);
	}
}

自定義LifecycleProcessor處理Lifecycle

在源碼分析中提到了DefaultLifecycleProcessor,其實(shí)現(xiàn)了LifecycleProcessor接口。然而我們自己也可以實(shí)現(xiàn)該接口,替換默認(rèn)的DefaultLifecycleProcessor。SpringBoot中則是自己配置了DefaultLifecycleProcessor,我們可以按照同樣的方式,覆蓋默認(rèn)的實(shí)現(xiàn)。例如可以讓Lifecycle中的start()方法在onRefresh()時(shí)也能被執(zhí)行。

org.springframework.boot.autoconfigure.context.LifecycleAutoConfiguration源碼:

/**
 * {@link EnableAutoConfiguration Auto-configuration} relating to the application
 * context's lifecycle.
 *
 * @author Andy Wilkinson
 * @since 2.3.0
 */
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(LifecycleProperties.class)
public class LifecycleAutoConfiguration {

	@Bean(name = AbstractApplicationContext.LIFECYCLE_PROCESSOR_BEAN_NAME)
	@ConditionalOnMissingBean(name = AbstractApplicationContext.LIFECYCLE_PROCESSOR_BEAN_NAME,
			search = SearchStrategy.CURRENT)
	public DefaultLifecycleProcessor defaultLifecycleProcessor(LifecycleProperties properties) {
		DefaultLifecycleProcessor lifecycleProcessor = new DefaultLifecycleProcessor();
		lifecycleProcessor.setTimeoutPerShutdownPhase(properties.getTimeoutPerShutdownPhase().toMillis());
		return lifecycleProcessor;
	}
}

SpringBoot中內(nèi)嵌web容器啟動(dòng)時(shí)機(jī)

SpringBoo中就是通過(guò)實(shí)現(xiàn)SmartLifecycle來(lái)啟動(dòng)內(nèi)嵌的web容器,實(shí)現(xiàn)類(lèi)為WebServerStartStopLifecycle。

ServletWebServerApplicationContextonRefresh方法中調(diào)用createWebServer,createWebServer方法中創(chuàng)建org.springframework.boot.web.server.WebServer實(shí)例,該對(duì)象則包含了控制web容器(tomcat、jetty)的啟動(dòng)與停止方法。

@Override
protected void onRefresh() {
	super.onRefresh();
	try {
		createWebServer();
	}catch (Throwable ex) {
		throw new ApplicationContextException("Unable to start web server", ex);
	}
}

ServletWebServerApplicationContext#createWebServer源碼:

private void createWebServer() {
	WebServer webServer = this.webServer;
	ServletContext servletContext = getServletContext();
	if (webServer == null && servletContext == null) {
		StartupStep createWebServer = this.getApplicationStartup().start("spring.boot.webserver.create");
		ServletWebServerFactory factory = getWebServerFactory();
		createWebServer.tag("factory", factory.getClass().toString());
		this.webServer = factory.getWebServer(getSelfInitializer());
		createWebServer.end();
		getBeanFactory().registerSingleton("webServerGracefulShutdown",
				new WebServerGracefulShutdownLifecycle(this.webServer));
		getBeanFactory().registerSingleton("webServerStartStop",
				new WebServerStartStopLifecycle(this, this.webServer));
	}
	else if (servletContext != null) {
		try {
			getSelfInitializer().onStartup(servletContext);
		}
		catch (ServletException ex) {
			throw new ApplicationContextException("Cannot initialize servlet context", ex);
		}
	}
	initPropertySources();
}

createWebServer方法會(huì)將創(chuàng)建的webServer封裝在WebServerStartStopLifecycle對(duì)象中,并注冊(cè)到Spring容器中。

org.springframework.boot.web.servlet.context.WebServerStartStopLifecycle源碼如下:

class WebServerStartStopLifecycle implements SmartLifecycle {

	private final ServletWebServerApplicationContext applicationContext;
	private final WebServer webServer;
	private volatile boolean running;

	WebServerStartStopLifecycle(ServletWebServerApplicationContext applicationContext, WebServer webServer) {
		this.applicationContext = applicationContext;
		this.webServer = webServer;
	}

	@Override
	public void start() {
		this.webServer.start();
		this.running = true;
		this.applicationContext
				.publishEvent(new ServletWebServerInitializedEvent(this.webServer, this.applicationContext));
	}

	@Override
	public void stop() { this.webServer.stop(); }

	@Override
	public boolean isRunning() {	return this.running; }

	@Override
	public int getPhase() { return Integer.MAX_VALUE - 1; }
}

WebServerStartStopLifecycle則實(shí)現(xiàn)了SmartLifecycle接口,當(dāng)Spring回調(diào)到SmartLifecycle接口方法時(shí)則調(diào)用this.webServer.start();啟動(dòng)web容器,web容器啟動(dòng)完成之后會(huì)通過(guò)applicationContext發(fā)布ServletWebServerInitializedEvent事件,表示web容器啟動(dòng)成功,可以接收http請(qǐng)求。

和SmartInitializingSingleton區(qū)別

相同點(diǎn):SmartInitializingSingletonLifecycle、SmartLifecycle都是在所有的單實(shí)例bean創(chuàng)建(getBean方法)之后執(zhí)行。

不同點(diǎn):

  • SmartInitializingSingleton優(yōu)先于Lifecycle、SmartLifecycle執(zhí)行。
  • SmartInitializingSingleton只有一個(gè)afterSingletonsInstantiated方法。而Lifecyclestart,stop,isRunning等方法。
  • 多個(gè)SmartInitializingSingleton實(shí)現(xiàn)之間無(wú)法排序控制執(zhí)行的順序,而SmartLifecycle實(shí)現(xiàn)了Phased接口,可以通過(guò)int getPhase()控制執(zhí)行循序。
  • SmartInitializingSingleton之間可以通過(guò)@DependsOn來(lái)控制執(zhí)行順序,但這是由Spring中@DependsOn注解的作用及原理來(lái)實(shí)現(xiàn)的. 并不是對(duì)SmartInitializingSingleton做了排序。

到此這篇關(guān)于Spring中SmartLifecycle和Lifecycle的作用和區(qū)別的文章就介紹到這了,更多相關(guān)Spring中SmartLifecycle和Lifecycle內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 解決MyEclipse中的Building workspace問(wèn)題的三個(gè)方法

    解決MyEclipse中的Building workspace問(wèn)題的三個(gè)方法

    這篇文章主要介紹了解決MyEclipse中的Building workspace問(wèn)題的三個(gè)方法,需要的朋友可以參考下
    2015-11-11
  • 基于swing實(shí)現(xiàn)窗體拖拽和拉伸

    基于swing實(shí)現(xiàn)窗體拖拽和拉伸

    這篇文章主要為大家詳細(xì)介紹了基于swing實(shí)現(xiàn)窗體拖拽和拉伸,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • 淺談幾種Java自定義異常處理方式

    淺談幾種Java自定義異常處理方式

    在Java中,異常是一種常見(jiàn)的處理機(jī)制,本文主要介紹了淺談幾種Java自定義異常處理方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05
  • Spring Boot Web 開(kāi)發(fā)注解篇

    Spring Boot Web 開(kāi)發(fā)注解篇

    在 Spring Boot 快速入門(mén)中,只要在 pom.xml 加入了 spring-boot-starter-web 依賴(lài),即可快速開(kāi)發(fā) web 應(yīng)用。下文給大家詳細(xì)介紹了spring boot web 開(kāi)發(fā)注解,感興趣的朋友參考下吧
    2017-08-08
  • Java使用poi生成word文檔的簡(jiǎn)單實(shí)例

    Java使用poi生成word文檔的簡(jiǎn)單實(shí)例

    Java POI是一個(gè)用于處理Microsoft Office文件(如Word、Excel和PowerPoint)的API,它是一個(gè)開(kāi)源庫(kù),允許Java開(kāi)發(fā)者讀取、創(chuàng)建和修改這些文檔,本文給大集介紹了Java使用poi生成word文檔的簡(jiǎn)單實(shí)例,感興趣的朋友可以參考下
    2024-06-06
  • Java并發(fā)編程中使用Executors類(lèi)創(chuàng)建和管理線(xiàn)程的用法

    Java并發(fā)編程中使用Executors類(lèi)創(chuàng)建和管理線(xiàn)程的用法

    這篇文章主要介紹了Java并發(fā)編程中使用Executors類(lèi)創(chuàng)建和管理線(xiàn)程的用法,文中舉了用其啟動(dòng)線(xiàn)程和設(shè)置線(xiàn)程優(yōu)先級(jí)的例子,需要的朋友可以參考下
    2016-03-03
  • Java實(shí)現(xiàn)HttpGet請(qǐng)求傳body參數(shù)

    Java實(shí)現(xiàn)HttpGet請(qǐng)求傳body參數(shù)

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)HttpGet請(qǐng)求傳body參數(shù)的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-02-02
  • MyBatis與Spring整合過(guò)程實(shí)例解析

    MyBatis與Spring整合過(guò)程實(shí)例解析

    這篇文章主要介紹了MyBatis與Spring整合過(guò)程實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-02-02
  • Java代碼審計(jì)之URL重定向的問(wèn)題解決

    Java代碼審計(jì)之URL重定向的問(wèn)題解決

    URLRedirect url重定向漏洞也稱(chēng)url任意跳轉(zhuǎn)漏洞,網(wǎng)站信任了用戶(hù)的輸入導(dǎo)致惡意攻擊,本文主要介紹了Java代碼審計(jì)之URL重定向的問(wèn)題解決,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-06-06
  • Mybatis-Plus中的條件參數(shù)使用

    Mybatis-Plus中的條件參數(shù)使用

    這篇文章主要介紹了Mybatis-Plus中的條件參數(shù)使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-05-05

最新評(píng)論