chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

還在使用kill -9 pid結(jié)束spring boot項目嗎?

Linux愛好者 ? 來源:CSDN技術(shù)社區(qū) ? 作者:流星007 ? 2021-04-13 16:01 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

kill可將指定的信息送至程序。預(yù)設(shè)的信息為SIGTERM(15),可將指定程序終止。若仍無法終止該程序,可使用SIGKILL(9)信息嘗試強(qiáng)制刪除程序。程序或工作的編號可利用ps指令或jobs指令查看(這段話來自菜鳥教程)。

講的這個復(fù)雜,簡單點來說就是用來殺死linux中的進(jìn)程,啥?你問我啥是進(jìn)程?請自行百度。

我相信很多人都用過kill -9 pid 這個命令,徹底殺死進(jìn)程的意思,一般情況我們使用它沒有上面問題,但是在我們項目中使用它就有可能存在致命的問題。

kill -9 pid 帶來的問題

由于kill -9 屬于暴力刪除,所以會給程序帶來比較嚴(yán)重的后果,那究竟會帶來什么后果呢?

舉個栗子:轉(zhuǎn)賬功能,再給兩個賬戶進(jìn)行加錢扣錢的時候突然斷電了?這個時候會發(fā)生什么事情?對于InnoDB存儲引擎來說,沒有什么損失,因為它支持事務(wù),但是對于MyISAM引擎來說那簡直就是災(zāi)難,為什么?假如給A賬戶扣了錢,現(xiàn)在需要將B賬戶加錢,這個時候停電了,就會造成,A的錢被扣了,但是B沒有拿到這筆錢,這在生產(chǎn)環(huán)境是絕對不允許的,kill -9 相當(dāng)于突然斷電的效果。

當(dāng)然了,像轉(zhuǎn)賬這種,肯定不是使用MyISAM引擎,但是如今分布式火了起來,跨服務(wù)轉(zhuǎn)賬已經(jīng)是很平常的事情,這種時候如果使用kill -9 去停止服務(wù),那就不是你的事務(wù)能保證數(shù)據(jù)的準(zhǔn)確性了,這個時候你可能會想到分布式事務(wù),這個世界上沒有絕對的安全系統(tǒng)或者架構(gòu),分布式事務(wù)也是一樣,他也會存在問題,概率很小,如果一旦發(fā)生,損失有可能是無法彌補(bǔ)的,所以一定不能使用kill -9 去停止服務(wù),因為你不知道他會造成什么后果。

在MyISAM引擎中表現(xiàn)的更明顯,比如用戶的信息由兩張表維護(hù),管理員修改用戶信息的時候需要修改兩張表,但由于你的kill -9 暴力結(jié)束項目,導(dǎo)致只修改成功了一張表,這也會導(dǎo)致數(shù)據(jù)的不一致性,這是小事,因為大不了再修改一次,但是金錢、合同這些重要的信息如果由于你的暴力刪除導(dǎo)致錯亂,我覺得可能比刪庫跑路還嚴(yán)重,至少刪庫還能恢復(fù),你這個都不知道錯在哪里。

那我們應(yīng)該怎么結(jié)束項目呢?

其實java給我們提供了結(jié)束項目的功能,比如:tomcat可以使用shutdown.bat/shutdown.sh進(jìn)行優(yōu)雅結(jié)束。

什么叫優(yōu)雅結(jié)束?

第一步:停止接收請求和內(nèi)部線程。

第二步:判斷是否有線程正在執(zhí)行。

第三步:等待正在執(zhí)行的線程執(zhí)行完畢。

第四步:停止容器。

以上四步才是正常的結(jié)束流程,那springboot怎么正常結(jié)束服務(wù)呢?下面我介紹幾種正常結(jié)束服務(wù)的方案,請拿好小本本做好筆記。

優(yōu)雅結(jié)束服務(wù)

kill -15 pid

這種方式也會比較優(yōu)雅的結(jié)束進(jìn)程(項目),使用他的時候需要慎重,為什么呢?我們來看個例子

我寫了一個普通的controller方法做測試

@GetMapping(value = “/test”) public String test(){ log.info(“test --- start”); try { Thread.sleep(100000); } catch (InterruptedException e) { e.printStackTrace(); } log.info(“test --- end”); return “test”; }

代碼很簡單,打?。簍est — start之后讓讓程序休眠100秒,然后再打印:test — end,在線程休眠中我們使用kill -15 pid來結(jié)束這個進(jìn)程,你們猜 test — end會被打印嗎?

application.yml

server: port: 9988

啟動項目

sudo mvn spring-boot:run

這是maven啟動springboot項目的方式

27269916-9b2f-11eb-8b86-12bb97331649.png

看到這個就代表項目啟動成了

找到項目的進(jìn)程id

sudo ps -ef |grep shutdown

27883e5a-9b2f-11eb-8b86-12bb97331649.png

這個就是項目的進(jìn)程號,接下來我們先測試test接口,讓線程進(jìn)入休眠狀態(tài),然后再使用kill -15 14086停止項目

sudo curl 127.0.0.1:9988/test

回到項目日志

27b44838-9b2f-11eb-8b86-12bb97331649.png

我們發(fā)現(xiàn)請求已經(jīng)到達(dá)服務(wù),并且線程已經(jīng)成功進(jìn)入休眠,現(xiàn)在我們kill -15 14086結(jié)束進(jìn)程

pIYBAGB1UQuActwnAASaBSQr8kc513.png

居然報錯了,但是test — end是打印出來了,為什么會報錯呢?這就和sleep這個方法有關(guān)了,在線程休眠期間,當(dāng)調(diào)用線程的interrupt方法的時候會導(dǎo)致sleep拋出異常,這里很明顯就是kill -15 這個命令會讓程序馬上調(diào)用線程的interrupt方法,目的是為了讓線程停止,雖然讓線程停止,但線程什么時候停止還是線程自己說的算,這就是為什么我們還能看到:test — end的原因。

ConfigurableApplicationContext colse

我們先看怎么實現(xiàn)

package com.ymy.controller; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.BeansException; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RestController; @RestController @Slf4j public class TestController implements ApplicationContextAware { private ApplicationContext context; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.context = applicationContext; } @GetMapping(value = “/test”) public String test(){ log.info(“test --- start”); try { Thread.sleep(100000); } catch (InterruptedException e) { e.printStackTrace(); } log.info(“test --- end”); return “test”; } /** * 停機(jī) */ @PostMapping(value = “shutdown”) public void shutdown(){ ConfigurableApplicationContext cyx = (ConfigurableApplicationContext) context; cyx.close(); } }

重點在:cyx.close();,為什么他能停止springboot項目呢?請看源碼

public void close() { synchronized(this.startupShutdownMonitor) { this.doClose(); if (this.shutdownHook != null) { try { Runtime.getRuntime().removeShutdownHook(this.shutdownHook); } catch (IllegalStateException var4) { } } } }

程序在啟動的時候向jvm注冊了一個關(guān)閉鉤子,我們在執(zhí)行colse方法的時候會刪除這個關(guān)閉鉤子,jvm就會知道這是需要停止服務(wù)。

我們看測試結(jié)果

282fec72-9b2f-11eb-8b86-12bb97331649.png

28ea8c3a-9b2f-11eb-8b86-12bb97331649.png

很明顯,他也出發(fā)了線程的interrupt方法導(dǎo)致線程報錯,原理和kill -15差不多。

actuator

這種方式是通過引入依賴的方式停止服務(wù),actuator提供了很多接口,比如健康檢查,基本信息等等,我們也可以使用他來優(yōu)雅的停機(jī)。

引入依賴

《dependency》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-actuator《/artifactId》 《/dependency》

application.yml

server: port: 9988 management: endpoints: web: exposure: include: shutdown endpoint: shutdown: enabled: true server: port: 8888

我這里對actuator的接口重新給定了一個接口,這樣可提高安全性,下面我們來測試一下

@RequestMapping(value = “/test”,method = RequestMethod.GET) public String test(){ System.out.println(“test --- start”); try { Thread.sleep(10000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(“test --- end”); return “hello”; }

在請求test途中停止服務(wù)

28f9bb6a-9b2f-11eb-8b86-12bb97331649.png

我們發(fā)現(xiàn)發(fā)送停止服務(wù)請求之后還給我們返回了提示信息,很人性化,我們看看控制臺

291be906-9b2f-11eb-8b86-12bb97331649.png

292f36aa-9b2f-11eb-8b86-12bb97331649.png

test — end被執(zhí)行了,不過在停止線程池的時候還是調(diào)用了線程的interrupt方法,導(dǎo)致sleep報錯,這三種方式都可以比較優(yōu)雅的停止springboot服務(wù),如果我項目中存在線程休眠,我希望10秒以后再停止服務(wù)可以嗎?肯定是可以的,我們只需要稍微做點修改就可以了。

1.新增停止springboot服務(wù)類:ElegantShutdownConfig.java

package com.ymy.config; import org.apache.catalina.connector.Connector; import org.springframework.boot.web.embedded.tomcat.TomcatConnectorCustomizer; import org.springframework.context.ApplicationListener; import org.springframework.context.event.ContextClosedEvent; import java.util.concurrent.Executor; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; public class ElegantShutdownConfig implements TomcatConnectorCustomizer, ApplicationListener《ContextClosedEvent》 { private volatile Connector connector; private final int waitTime = 10; @Override public void customize(Connector connector) { this.connector = connector; } @Override public void onApplicationEvent(ContextClosedEvent event) { connector.pause(); Executor executor = connector.getProtocolHandler().getExecutor(); if (executor instanceof ThreadPoolExecutor) { try { ThreadPoolExecutor threadPoolExecutor = (ThreadPoolExecutor) executor; threadPoolExecutor.shutdown(); if (!threadPoolExecutor.awaitTermination(waitTime, TimeUnit.SECONDS)) { System.out.println(“請嘗試暴力關(guān)閉”); } } catch (InterruptedException ex) { System.out.println(“異常了”); Thread.currentThread().interrupt(); } } } }

2.在啟動類中加入bean

package com.ymy; import com.ymy.config.ElegantShutdownConfig; import lombok.extern.slf4j.Slf4j; import org.apache.catalina.connector.Connector; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.web.embedded.tomcat.TomcatConnectorCustomizer; import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory; import org.springframework.boot.web.servlet.server.ServletWebServerFactory; import org.springframework.context.ApplicationListener; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.event.ContextClosedEvent; import java.util.concurrent.Executor; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; @SpringBootApplication public class ShutdownServerApplication { public static void main(String[] args) { ConfigurableApplicationContext run = SpringApplication.run(ShutdownServerApplication.class, args); run.registerShutdownHook(); } @Bean public ElegantShutdownConfig elegantShutdownConfig() { return new ElegantShutdownConfig(); } @Bean public ServletWebServerFactory servletContainer() { TomcatServletWebServerFactory tomcat = new TomcatServletWebServerFactory(); tomcat.addConnectorCustomizers(elegantShutdownConfig()); return tomcat; } }

這樣我們就配置好了,我們再來測試一遍,test的接口還是休眠10秒

2961f3c4-9b2f-11eb-8b86-12bb97331649.png

我們發(fā)現(xiàn)這次沒有報錯了,他是等待了一段時間之后再結(jié)束的線程池,這個時間就是我們在ElegantShutdownConfig類中配置的waitTime。

那可能你會有疑問了,jvm沒有立即停止,那這個時候在有請求會發(fā)生什么呢?如果關(guān)閉的時候有新的請求,服務(wù)將不在接收此請求。

數(shù)據(jù)備份操作

如果我想在服務(wù)停止的時候做點備份操作啥的,應(yīng)該怎么做呢?其實很簡單在你要執(zhí)行的方法上添加一個注解即可:@PreDestroy

Destroy:消滅、毀滅

pre:前綴縮寫

所以合在一起的意思就是在容器停止之前執(zhí)行一次,你可以在這里面做備份操作,也可以做記錄停機(jī)時間等。

新增服務(wù)停止備份工具類:DataBackupConfig.java

package com.ymy.config; import org.springframework.context.annotation.Configuration; import javax.annotation.PreDestroy; @Configuration public class DataBackupConfig { @PreDestroy public void backData(){ System.out.println(“正在備份數(shù)據(jù)。。。。。。。。。。?!保? } }

我們再來測試然后打印控制臺日志:

29a04a16-9b2f-11eb-8b86-12bb97331649.png

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • JAVA
    +關(guān)注

    關(guān)注

    20

    文章

    2997

    瀏覽量

    115606
  • SAM
    SAM
    +關(guān)注

    關(guān)注

    0

    文章

    115

    瀏覽量

    34296
  • kill
    +關(guān)注

    關(guān)注

    0

    文章

    9

    瀏覽量

    2246

原文標(biāo)題:還在使用kill -9 pid結(jié)束spring boot項目嗎?不妨試試這幾種優(yōu)雅的方式!

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點推薦

    深入理解?RK3506 U-Boot?重定位:從代碼到原理

    在嵌入式系統(tǒng)中,U-Boot?作為引導(dǎo)加載程序,其啟動流程的核心環(huán)節(jié)之一就是 重定位(Relocation) 。對于?RK3506?這類基于?ARM Cortex-A?架構(gòu)的芯片,重定位的本質(zhì)是將
    的頭像 發(fā)表于 11-28 07:05 ?33次閱讀
    深入理解?RK3506 U-<b class='flag-5'>Boot</b>?重定位:從代碼到原理

    一款基于Java+Spring Boot+Vue的智慧隨訪管理系統(tǒng)源碼

    智慧隨訪管理系統(tǒng)源碼,一款基于Java+Spring Boot+Vue的B/S架構(gòu)醫(yī)院隨訪管理系統(tǒng)源碼,采用前后端分離技術(shù)(Ant-Design+MySQL5),具有自主版權(quán)和落地案例。 隨訪管理
    的頭像 發(fā)表于 11-13 15:38 ?169次閱讀
    一款基于Java+<b class='flag-5'>Spring</b> <b class='flag-5'>Boot</b>+Vue的智慧隨訪管理系統(tǒng)源碼

    飛凌嵌入式ElfBoard-Linux系統(tǒng)基礎(chǔ)入門-其它shell命令

    強(qiáng)制終止進(jìn)程命令命令:kill功能:強(qiáng)制終止進(jìn)程參數(shù):-signal pid終止進(jìn)程之前終止進(jìn)110746程之后,kill -9表示無條件殺死該進(jìn)程,殺死進(jìn)程后終端窗口會關(guān)閉:elf@
    發(fā)表于 10-11 08:42

    PID控制算法學(xué)習(xí)筆記資料

    用于新手學(xué)習(xí)PID控制算法。
    發(fā)表于 08-12 16:22 ?7次下載

    Spring攔截器:你的請求休想逃過我的五指山!

    Spring框架中,攔截器(Interceptor)是一種強(qiáng)大的機(jī)制,它允許開發(fā)者在請求處理的不同階段插入自定義邏輯。WebApplicationContext作為Spring Web應(yīng)用的上下文容器,為攔截器的配置和管理提供了基礎(chǔ)支持。
    的頭像 發(fā)表于 07-26 11:25 ?505次閱讀
    <b class='flag-5'>Spring</b>攔截器:你的請求休想逃過我的五指山!

    IAP升級跳轉(zhuǎn)APP跳轉(zhuǎn)Boot失敗了怎么解決?

    使用的MCU是STM32U535RC ,V6 編譯器 跳轉(zhuǎn)實現(xiàn)的功能如下 /** *@brief直接跳轉(zhuǎn)到 Boot 程序 */ void LibJumpToBoot(UINT32
    發(fā)表于 07-21 06:15

    瑞薩RA芯片的Boot模式簡介

    RA芯片在上電或通過芯片復(fù)位引腳進(jìn)行復(fù)位時,會根據(jù)MD引腳的電平來進(jìn)入不同的芯片操作模式:“Single-chip Mode”或者“Boot Mode”。
    的頭像 發(fā)表于 04-09 10:52 ?1922次閱讀
    瑞薩RA芯片的<b class='flag-5'>Boot</b>模式簡介

    如何在NXP MCU Boot Utility中對電路板進(jìn)行檢測?

    、2-OFF、3-OFF、4-ON(串行下載器編程模式),并且還在 J86 中連接 USB。 并附上(問題的屏幕截圖)文件。 我需要如何在 NXP MCU Boot Utility 中對電路板進(jìn)行檢測?
    發(fā)表于 03-17 06:33

    限時免積分下載:增量式與位置式PID算法的C語言實現(xiàn)分享

    前面咱們有分享對PID算法離散化和增量式PID算法原理進(jìn)行來探索,之后又使用Matlab進(jìn)行了仿真實驗,對PID三個參數(shù)又有了更深入的認(rèn)識,接下來我們來使用C語言進(jìn)行PID算法實現(xiàn),并
    發(fā)表于 03-05 18:32

    PID控制詳解(可下載)

    一、PID控制簡介 PID( Proportional Integral Derivative)控制是最早發(fā)展起來的控制策略之一,由于其算法簡單、魯棒 性好和可靠性高,被廣泛應(yīng)用于工業(yè)過程
    發(fā)表于 02-27 15:42 ?5次下載

    PID發(fā)展趨勢分析

    摘要:文檔中簡要回顧了 PID 控制器的發(fā)展歷程,綜述了 PID 控制的基礎(chǔ)理論。對 PID 控制今后的發(fā)展進(jìn)行了展望。重點介紹了比例、積分、微分基本控制規(guī)律,及其優(yōu)、缺點。關(guān)鍵詞:PID
    發(fā)表于 02-26 15:27

    校園點餐訂餐外賣跑腿Java源碼

    創(chuàng)建一個校園點餐訂餐外賣跑腿系統(tǒng)是一個復(fù)雜的項目,涉及到前端、后端、數(shù)據(jù)庫設(shè)計等多個方面。在這里,我可以提供一個簡化的Java后端示例,使用Spring Boot框架來搭建一個基本的API服務(wù)。這個
    的頭像 發(fā)表于 12-24 14:55 ?909次閱讀
    校園點餐訂餐外賣跑腿Java源碼

    寶藏級微服務(wù)架構(gòu)工具合集

    寶藏級熱門微服務(wù)架構(gòu)工具包含Spring Boot、Eclipse Vert.X、Kubernetes、Tyk、RabbitMQ、Apache Kafka等。其中,Spring Boot
    的頭像 發(fā)表于 12-21 16:33 ?881次閱讀

    SSM開發(fā)環(huán)境的搭建教程 SSM與Spring Boot的區(qū)別

    SSM開發(fā)環(huán)境的搭建教程 SSM(Spring+SpringMVC+MyBatis)開發(fā)環(huán)境的搭建涉及多個步驟,以下是詳細(xì)的教程: 創(chuàng)建Maven項目 : 使用Maven工具創(chuàng)建一個新的Maven
    的頭像 發(fā)表于 12-16 18:13 ?1799次閱讀

    Spring 應(yīng)用合并之路(二):峰回路轉(zhuǎn),柳暗花明

    提醒下,決定拋開 Spring Boot 內(nèi)置的父子容器方案,完全自己實現(xiàn)父子容器。 如何加載 web 項目? 現(xiàn)在的難題只有一個:如何加載 web 項目?加載完成后,如何持續(xù)持有 w
    的頭像 發(fā)表于 12-12 11:22 ?1395次閱讀