Java應用中使用ShutdownHook友好地清理現場


  在線上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.netease.test.java.lang;

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");

    //計數干活次數
    static AtomicInteger count = new AtomicInteger(0);

    /**
     * hook線程
     */
    static class CleanWorkThread extends 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();
            }
        }
    }
    public static void main(String[] args) throws InterruptedException {
        //將hook線程添加到運行時環境中去
        Runtime.getRuntime().addShutdownHook(new CleanWorkThread());
        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);

    }
}

 



運行后,可以模擬以上五種場景進行測試,只有kill -9 pid不會執行Hook里面的代碼。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM