關於jvm鉤子 Runtime.getRuntime().addShutdownHook
阿新 • • 發佈:2017-08-11
illegal tex stack 一個 聲明 [] time() 被調用 base
轉自:
http://www.cnblogs.com/nexiyi/p/java_add_ShutdownHook.html
在線上Java程序中經常遇到進程程掛掉,一些狀態沒有正確的保存下來,這時候就需要在JVM關掉的時候執行一些清理現場的代碼。Java中得ShutdownHook提供了比較好的方案。
JDK在1.3之後提供了Java Runtime.addShutdownHook(Thread hook)方法,可以註冊一個JVM關閉的鉤子,這個鉤子可以在以下幾種場景被調用:
- 1)程序正常退出
- 2)使用System.exit()
- 3)終端使用Ctrl+C觸發的中斷
- 4)系統關閉
- 5)使用Kill pid命令幹掉進程
註:在使用kill -9 pid是不會JVM註冊的鉤子不會被調用。
在JDK中方法的聲明:
public void addShutdownHook(Thread hook)
參數
hook -- 一個初始化但尚未啟動的線程對象,註冊到JVM鉤子的運行代碼。
異常
IllegalArgumentException -- 如果指定的鉤已被註冊,或如果它可以判定鉤已經運行或已被運行
IllegalStateException -- 如果虛擬機已經是在關閉的過程中
SecurityException -- 如果存在安全管理器並且它拒絕的RuntimePermission(“shutdownHooks”)
代碼示例:
使用Timer模擬一個工作線程,該線程重復工作十次,使用System.exit()退出,在清理現場代碼CleanWorkThread 中,取消timer運行,並輸出必要的日誌信息。
package com.yun.base.mq.test; import java.util.Timer; import java.util.TimerTask; import java.util.concurrent.atomic.AtomicInteger; /** * Date: 14-6-18 * Time: 11:01 * 測試ShutdownHook */ public class TestShutdownHook { //簡單模擬幹活的 static Timer timer = new Timer("job-timer"); //計數幹活次數 staticAtomicInteger count = new AtomicInteger(0); public static void main(String[] args) throws InterruptedException { //將hook線程添加到運行時環境中去 Runtime.getRuntime().addShutdownHook(new Thread(){ @Override public void run() { System.out.println("clean some work."); timer.cancel(); try { Thread.sleep(2 * 1000);//sleep 2s } catch (InterruptedException e) { e.printStackTrace(); } } }); System.out.println("main class start ..... "); //簡單模擬 timer.schedule(new TimerTask() { @Override public void run() { count.getAndIncrement(); System.out.println("doing job " + count); if (count.get() == 10) { //幹了10次退出 System.exit(0); } } }, 0, 2 * 1000); } }
每2秒執行一次工作線程,10次之後關閉系統,啟用鉤子關閉工作線程。
關於jvm鉤子 Runtime.getRuntime().addShutdownHook