国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

不懂優雅停機,搞掛了線上服務,咋辦?

系統 Linux
在 Linux 世界里,一切都是資源。當我們啟動一個 JVM 的時候,我們就加載了許多的資源。而當我們關閉 JVM 的時候,JVM 只會釋放內存這個資源,而其他資源是不會釋放的,例如:網絡連接、文件句柄等等。

公司項目是用 consul 進行注冊的,在發布微服務的時候,總是會導致調用方出現一定幾率的調用失敗。一開始百思不得其解,后來咨詢了資深的同事才知道:原來是服務下線的時候沒有優雅停機,沒有去 consul 將自己下線再停機,導致調用方拿到了舊的調用地址,導致調用失敗! 看來優雅停機還是一個蠻重要的知識點,可不能忽略,今天就讓我們來盤盤它吧!

什么是優雅停機?

在 Linux 世界里,一切都是資源。當我們啟動一個 JVM 的時候,我們就加載了許多的資源。而當我們關閉 JVM 的時候,JVM 只會釋放內存這個資源,而其他資源是不會釋放的,例如:網絡連接、文件句柄等等。

Linux 的網絡連接數、文件句柄數都是有限的,如果我們沒有及時釋放,時間久了就會導致一些奇怪的問題。那么如何在 JVM 關閉的時候,釋放這些資源呢?答案就是:利用 Java 提供的 ShutdownHook 接口。 我們所說的優雅停機,就是利用 Java 提供的 ShutdownHook 接口注冊一個鉤子,讓 JVM 在關閉之前執行鉤子函數的代碼,讓其關閉對應的資源。

適用場景

在學會怎么使用優雅停機之前,我們需要弄清楚優雅停機適用于哪些場景,那我們就需要先弄清楚 JVM 關閉的幾種情況了。JVM 關閉的情況可以分為 3 大類 11 個情況,如下圖所示:

圖片

JVM 關閉的場景

在 JVM 關閉的 3 大類場景中,只有正常關閉與異常關閉是支持優雅停機的,而強制關閉則是不支持的。下面我們通過三個例子來驗證一下。

JVM 正常關閉

JVM 正常關閉這種情況,我們只需要正常運行一個 main 函數,然后為其注冊一個 ShutdownHook 即可,其代碼如下所示。

public class NormalShutdownTest {
public void start() {
Runtime.getRuntime().addShutdownHook(new Thread(() ->
System.out.println("鉤子函數被執行,可以在這里關閉資源。")
));
}

public static void main(String[] args) {
new NormalShutdownTest().start();
System.out.println("主應用程序在執行,正常關閉。");
}
}

輸出結果為:

主應用程序在執行,正常關閉。
鉤子函數被執行,可以在這里關閉資源。

可以看到鉤子函數的代碼正常執行了。如果你在 main 函數增加 System.exit(0) 代碼,執行之后的結果也還是一樣。這說明 JVM 正常關閉情況下,是支持優雅停機的。

異常關閉

JVM 異常關閉這種情況,我們嘗試制造內存溢出。只需要聲明一個 500 MB 的數組,然后設置 JVM 堆最大為 20 MB 即可(-Xmx20M),其代碼如下所示。

public class OomShutdownTest {
public void start() {
Runtime.getRuntime().addShutdownHook(new Thread(() ->
System.out.println("鉤子函數被執行,可以在這里關閉資源")
));
}

public static void main(String[] args) throws Exception {
new OomShutdownTest().start();
System.out.println("主應用程序在執行,內存溢出關閉。");
byte[] b = new byte[500 * 1024 * 1024];
}
}

執行結果為:

主應用程序在執行,內存溢出關閉。
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at tech.shuyi.javacodechip.shutdownhook.OomShutdownTest.main(OomShutdownTest.java:13)
鉤子函數被執行,可以在這里關閉資源

可以看到 JVM 拋出了 OOM 錯誤,但是鉤子函數還是被執行了。如果你在 main 函數中自行拋出 RuntimeException,鉤子函數也還是會被執行。感興趣的朋友可以自行嘗試一下。

強制關閉

JVM 強制關閉這種情況,我們可以使用 Runtime.getRuntime().halt(1) 進行測試,其代碼如下所示。

public class ForceShutdownTest {
public void start() {
Runtime.getRuntime().addShutdownHook(new Thread(() ->
System.out.println("鉤子函數被執行,可以在這里關閉資源。")
));
}

public static void main(String[] args) throws Exception {
new ForceShutdownTest().start();
System.out.println("主應用程序在執行,強制關閉。");
Runtime.getRuntime().halt(1);
}
}

執行結果:

主應用程序在執行,強制關閉。

可以看到鉤子函數并沒有被執行,所以 JVM 強制關閉這種場景不支持優雅停機。

最佳實踐

看了上面的例子,看起來優雅停機沒那么復雜嘛。實際上,優雅停機用不好,很可能出現一些其他問題。這里給出幾個最佳實踐原則,幫助大家用好優雅停機!

只注冊一個鉤子

我們都知道 JVM 可以注冊多個鉤子,而鉤子本質上是一個線程,可以并發執行。那么就很可能出現鉤子之間相互依賴,這樣就會導致依賴死鎖了。另外,也可能因為多個鉤子操作同一個資源,導致資源競爭出現死鎖。因此,較好的一種方式就是只注冊一個鉤子,所有的資源釋放都在這個鉤子中操作。

確保線程安全

因為鉤子本質上也是一個線程,JVM 可能會并發執行多個鉤子,JVM 并不保證它們的執行順序,因此需要保證鉤子中的操作是線程安全的。當然了,如果你只有一個鉤子的話,那這個提示可以忽略了。

不要做耗時的操作

在鉤子中,不要做耗時的操作。因為當我們要關閉 JVM 時,用戶肯定是希望盡快關閉,因此鉤子中主要用于關閉殘留資源,不應該再做其他耗時的操作。

不要做注冊、移除鉤子的操作

在關閉鉤子中,不能執行注冊、移除鉤子的操作,否則 JVM 拋出 IllegalStateException。

不要調用 System.exit () 操作

也不能調用 System.exit() 操作,但是調用 Runtime.halt() 操作是可以的。我想,這是因為調用 System.exit () 操作會導致循環進入鉤子,導致死循環吧。

需要考慮的資源

除了上面一些代碼上的操作需要考慮,我們還需要注意下面這些場景的處理:

  • 池化資源的釋放:數據庫連接池、HTTP 連接池、線程池。
  • 在處理線程的釋放:已經被連接的 HTTP 請求。
  • MQ 消費者的處理:正在處理的消息。
  • 隱形受影響的資源的處理:Zookeeper、Nacos 實例下線等。

應用案例

Java 提供的優雅停機機制,可以說是許多框架的基礎。諸如 Spring、Consul 等中間件框架,都是利用 Java 提供的這個機制進行優雅停機的。

Spring 的優雅停機

例如 Spring 是基于 Java 語言開發的框架,那其也勢必依賴于 JVM 的 ShutdownHook。Spring 關于優雅停機的代碼在 org.springframework.context.support.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();
}
}
};
// 增加 ShutdownHook 鉤子
Runtime.getRuntime().addShutdownHook(this.shutdownHook);
}
}

可以看到 Spring 在 registerShutdownHook() 函數里,注冊了一個關閉的鉤子,鉤子中調用了 doClose() 方法。

服務治理的優雅停機

不論是 Dubbo 還是 Spring Cloud 的分布式服務框架,需要關注的是怎么能在服務停止前,先將提供者在注冊中心進行反注冊,然后在停止服務提供者,這樣才能保證業務系統不會產生各種 503、timeout 等現象。為了實現上述說到的效果,那么我們就必須關注優雅停機這件事情。

彩蛋

我們都知道通過 kill -15 可以讓 JVM 優雅停機,那我們是否可以監聽特定的信號量,從而讓程序做特定的操作呢?例如:讓 JVM 監聽第 12 信號量,然后打印一條日志,隨后優雅停機。

答案是當然可以啦!我們只需要利用 Signal 類,并實現一個 SignHandler 類就可以了。其實現代碼如下所示:

public class CustomShutdownTest {
public void start() {
Runtime.getRuntime().addShutdownHook(new Thread(() ->
System.out.println("鉤子函數被執行,可以在這里關閉資源。")
));
}

public static void main(String[] args) {
// custom signal kill
Signal sg = new Signal("USR2"); // kill -12 pid
Signal.handle(sg, new SignalHandler() {
@Override
public void handle(Signal signal) {
System.out.println("接收到信號量:" + signal.getName());
// 監聽信號量,通過System.exit(0)正常關閉JVM,觸發關閉鉤子執行收尾工作
System.exit(0);
}
});
// other logic
new CustomShutdownTest().start();
System.out.println("主應用程序在執行,正常關閉。");
try {
Thread.sleep(30000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

我們啟動該類后,先讓其休眠 30 秒,隨后用 jps 命令找到進程 ID,隨后運行 kill -USR2 PID 即可,如截圖所示。

圖片

隨后可以看到控制臺打印出如下消息:

主應用程序在執行,正常關閉。
接收到信號量:USR2
鉤子函數被執行,可以在這里關閉資源。

從上面消息我們知道,JVM 成功接收到了 USR2 信號量,也成功執行了鉤子函數。搞定!

提示:其實 USR2 是 Linux 第 12 個信號量,是留給用戶使用的一個信號量。我們可以通過該信號量做一些定制化操作,從而實現更加復雜的功能。

參考資料

  • 如何優雅地停止 Java 進程 | iBit 程序猿
  • JVM 進程的優雅關閉 - waterystone - 博客園
  • VIP!例子比較不錯!如何優雅的關閉 JVM?-51CTO.COM
  • 比較深度一些,范圍比較廣!Spring—— 項目優雅停機 - 曹偉雄 - 博客園
  • RPC 服務治理相關。VIP!研究優雅停機時的一點思考 | 徐靖峰 | 個人博客
  • Spring 的優雅停機 | Yanick's Blog
  • rocketmq 優雅停機往事 - 云 + 社區 - 騰訊云?
責任編輯:武曉燕 來源: 陳樹義
相關推薦

2022-07-24 09:46:48

優雅停機代碼

2025-07-29 08:25:57

2021-05-08 08:33:00

Rocketmq日志數據源

2021-04-19 09:37:12

RocketMQ集群版本

2023-01-30 07:41:43

2020-08-25 10:34:22

微服務微服務架構生產環境

2025-03-17 00:00:00

2024-03-18 14:06:00

停機Spring服務器

2020-08-03 07:04:54

測試面試官應用程序

2021-08-02 15:06:46

vim服務Java

2025-03-11 00:55:00

Spring停機安全

2011-11-18 16:41:42

IDC世紀互聯

2021-02-06 13:11:28

SQL系統數據庫

2021-06-10 06:59:34

Redis應用API

2019-10-18 15:07:14

數據科學職業生涯萬努克

2020-10-14 10:29:58

人工智能

2021-10-03 15:00:44

數據庫mysql單機

2018-11-01 13:38:51

Java中斷停止

2024-07-22 19:31:34

2020-11-18 08:17:14

Java源碼Class
點贊
收藏

51CTO技術棧公眾號

精品国产福利| 国产精品免费一区豆花| 宅男噜噜噜66一区二区66| 涩涩涩999| 你懂的国产精品永久在线| 色综合色综合网色综合| 在线国产成人影院| 欧美美女视频在线观看| 国产精品99久久久久久www| 91精品韩国| 精品国产一区二区三区久久久蜜月 | 亚洲精品不卡在线| 亚洲精品永久免费| 最新av在线免费观看| 国内精品久久久久久久影视麻豆 | 免费一级特黄特色毛片久久看| 狠狠做六月爱婷婷综合aⅴ| 不卡av在线播放| 日韩av综合| 日本一区二区三级电影在线观看| 国内自拍中文字幕| 影音先锋欧美激情| 美女久久久久久久| 色老板在线视频| 中文字幕亚洲一区二区va在线| 777777av| 国产精品超碰97尤物18| 国产精品自拍视频在线| 国产精品久久久久久久| 国产区精品视频| 午夜日韩电影| 国产女教师bbwbbwbbw| 亚洲福利视频网| 亚洲а∨天堂久久精品喷水| 成年女人的天堂在线| 欧美性猛片xxxx免费看久爱| 成人在线免费电影| 欧美美女一区二区| 成人激情视频在线播放| 久久免费一区| 亚洲九九精品| 国产丝袜精品第一页| 欧洲一区精品| 91色porny在线视频| 毛片毛片毛片毛片毛片毛片毛片毛片毛片| 91啦中文在线观看| 欧美日韩夜夜| 亚洲欧美激情精品一区二区| 久久激情视频久久| 亚洲精品影片| 国产精品久久久久久一区二区| 国产一区二区三区四区二区| 国产精品免费在线免费| 日本欧美在线观看| 99re热视频在线| 美脚恋feet久草欧美| 亚洲天堂av在线免费观看| 九色自拍视频在线观看| 国产精品天美传媒沈樵| 国产精品视频二区三区| 亚洲性线免费观看视频成熟| 8×8x拔擦拔擦在线视频网站| 日韩国产高清在线| 成人观看视频| 超碰精品一区二区三区乱码| 日本中文一区二区三区| 青娱在线视频| 5278欧美一区二区三区| 国产精品资源在线观看| 91久久精品国产91久久| 96av麻豆蜜桃一区二区| 2024最新电影免费在线观看| 国产精品成人一区二区三区吃奶| 久久久久久日产精品| 极品粉嫩国产18尤物| 日韩av最新在线观看| 亚洲日韩视频| 免费中文日韩| 久久国产直播| 美洲精品一卡2卡三卡4卡四卡| 成人免费aaa| 国产欧美日韩91| 亚洲国产精品va在线看黑人| 91视频你懂的| 中文日韩欧美| 国产无遮挡裸体免费久久| 亚洲一区二区三区视频| 日韩精品一区二区在线观看| 国产视频一区二区在线| 日本三级韩国三级欧美三级| 热久久精品免费视频| 久久精品第九区免费观看| 欧美大片在线影院| 亚洲v精品v日韩v欧美v专区| 成人免费高清观看| 久久九九国产视频| 鲁片一区二区三区| 日产精品久久久一区二区福利| 日韩av在线不卡| 在线观看亚洲一区| 亚洲日本在线观看| 国产suv精品一区二区三区| 欧美高清日韩| 精品国产乱码久久久| 三级精品视频| 成人51免费| 国产夫妻在线播放| 国产一级二级三级在线观看| 羞羞在线视频| 日本免费视频| 福利小视频在线观看| 天堂а√在线官网| 黑人精品视频| 成人免费在线观看视频| 亚洲欧洲精品一区二区三区波多野1战4 | 自拍偷拍 国产| 国产精品theporn88| 久久99精品国产麻豆不卡| 日韩av网址大全| 97超碰免费在线| 美女网站在线| 热久久精品免费视频| 久久久一本精品99久久精品66| 日韩av手机在线| 欧美日本黄视频| 精品国产乱码久久久久久图片 | 欧美成人a∨高清免费观看| 日韩中文字幕无砖| 国产成人高清精品| 91久久精品在线| 久久免费视频观看| 国产日韩欧美在线播放不卡| 亚洲性视频在线| 97人人做人人爽香蕉精品| 国产日产一区二区| 91色视频在线导航| 欧美在线一级va免费观看| 日韩在线高清视频| 亚洲日本中文字幕| 中文字幕无线精品亚洲乱码一区 | 国产1区在线| 久久av色综合| 久草在线新免费首页资源站| 黄色在线视频网站| 在线观看中文| 久久男人天堂| 一级毛片久久久| 亚洲精品555| 999www成人| 黄页在线播放| 福利视频网站| 美女毛片在线看| 高清日韩av电影| 夜鲁夜鲁夜鲁视频在线播放| 欧美日韩123区| 久久视频免费在线| 91动漫在线看| 97视频在线观看亚洲| 欧美xxxx做受欧美.88| 国产精品欧美在线| 欧美婷婷久久| 日本www高清视频| 久久mv成人精品亚洲动漫| 日韩免费在线看| 国产精品成av人在线视午夜片| 欧美一区二区视频97| 97视频在线观看播放| 成人女人免费毛片| 亚洲国产欧美一区二区三区不卡| av免费网站观看| 中文字幕大看焦在线看| 高清一区二区三区视频| avove在线观看| 国产成人+综合亚洲+天堂| 国产精品一区而去| 窝窝九色成人影院| 1769在线观看| 2020最新国产精品| 亚洲一区二区三区无吗| 新狼窝色av性久久久久久| 国产极品在线观看| 久久精品国产亚洲blacked| 男女视频在线| 一区二区三区国产好| 日韩午夜免费视频| 中文欧美字幕免费| 精品国产一区二区国模嫣然| 清纯唯美日韩制服另类| 亚洲国产一区在线| 性网站在线看| 99久久人爽人人添人人澡| 精品999网站| 一区二区三区在线视频观看| 精品伊人久久久久7777人| 欧美性猛交xxxx免费看漫画| 欧美美女15p| 特级西西444www大精品视频| 一区二区高清不卡| 欧美1区2区3区| 亚洲一区在线观看免费观看电影高清 |