偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java并發(fā)編程:優(yōu)雅的關(guān)閉鉤子(Shutdown Hook)

開發(fā) 前端
當(dāng)程序即將退出時(shí)(例如釋放資源、關(guān)閉數(shù)據(jù)庫(kù)連接等),可以通過(guò)預(yù)先注冊(cè)一個(gè)或多個(gè)關(guān)閉鉤子線程(Shutdown Hook)來(lái)執(zhí)行相關(guān)操作。當(dāng) JVM 進(jìn)程準(zhǔn)備退出時(shí),這些鉤子線程會(huì)被觸發(fā)并運(yùn)行。

關(guān)閉鉤子簡(jiǎn)介

當(dāng)程序即將退出時(shí)(例如釋放資源、關(guān)閉數(shù)據(jù)庫(kù)連接等),可以通過(guò)預(yù)先注冊(cè)一個(gè)或多個(gè)關(guān)閉鉤子線程(Shutdown Hook)來(lái)執(zhí)行相關(guān)操作。當(dāng) JVM 進(jìn)程準(zhǔn)備退出時(shí),這些鉤子線程會(huì)被觸發(fā)并運(yùn)行。

示例代碼:

public class HookThreadDemo {
    privatestaticclass HookRunnable implements Runnable {
        @Override
        public void run() {
            try {
                System.out.println("鉤子線程 " + Thread.currentThread().getName() + " 正在執(zhí)行...");
                Thread.sleep(2000L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("鉤子線程 " + Thread.currentThread().getName() + " 執(zhí)行結(jié)束");
        }
    }

    public static void main(String[] args) {
        HookRunnable hookRunnable = new HookRunnable();
        // 添加鉤子線程 0
        Runtime.getRuntime().addShutdownHook(new Thread(hookRunnable));
        // 添加鉤子線程 1
        Runtime.getRuntime().addShutdownHook(new Thread(hookRunnable));
        System.out.println("主線程即將結(jié)束執(zhí)行");
    }
}

輸出結(jié)果:

主線程即將結(jié)束執(zhí)行
鉤子線程 Thread-0 正在執(zhí)行...
鉤子線程 Thread-1 正在執(zhí)行...
鉤子線程 Thread-1 執(zhí)行結(jié)束
鉤子線程 Thread-0 執(zhí)行結(jié)束

當(dāng)主線程執(zhí)行完畢后,JVM 進(jìn)程退出前,所有注冊(cè)的鉤子線程會(huì)被啟動(dòng)并執(zhí)行。

關(guān)閉鉤子應(yīng)用場(chǎng)景

  1. 釋放資源:關(guān)閉文件句柄、數(shù)據(jù)庫(kù)連接等,避免資源泄漏。
  2. 停止服務(wù):安全關(guān)閉服務(wù)器,確保所有請(qǐng)求處理完畢。
  3. 發(fā)送通知:通過(guò)郵件或短信通知用戶服務(wù)已停止。
  4. 記錄日志:保存系統(tǒng)狀態(tài)或錯(cuò)誤信息,便于后續(xù)排查問(wèn)題。

數(shù)據(jù)庫(kù)連接實(shí)戰(zhàn)演示

以下代碼演示如何用關(guān)閉鉤子關(guān)閉數(shù)據(jù)庫(kù)連接:

public class DatabaseConnection {
    privatestatic Connection conn;

    public static void main(String[] args) {
        System.out.println("主線程開始執(zhí)行");
        initConnection(); // 初始化數(shù)據(jù)庫(kù)連接
        System.out.println("執(zhí)行數(shù)據(jù)查詢與處理");

        // 注冊(cè)關(guān)閉鉤子
        Runtime.getRuntime().addShutdownHook(new Thread(() -> closeConnection()));
        System.out.println("主線程結(jié)束執(zhí)行");
    }

    private static void initConnection() {
        try {
            Class.forName("com.mysql.cj.jdbc.Driver");
            conn = DriverManager.getConnection(
                "jdbc:mysql://localhost:3306/school_info?useSSL=true&",
                "root", "root"
            );
            System.out.println("數(shù)據(jù)庫(kù)連接成功!");
        } catch (ClassNotFoundException | SQLException e) {
            e.printStackTrace();
        }
    }

    private static void closeConnection() {
        try {
            conn.close();
            System.out.println("數(shù)據(jù)庫(kù)連接已關(guān)閉!");
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

輸出結(jié)果:

主線程開始執(zhí)行
數(shù)據(jù)庫(kù)連接成功!
執(zhí)行數(shù)據(jù)查詢與處理
主線程結(jié)束執(zhí)行
數(shù)據(jù)庫(kù)連接已關(guān)閉!

使用關(guān)閉鉤子的注意事項(xiàng)

  1. 強(qiáng)制終止進(jìn)程(如kill -9)不會(huì)觸發(fā)鉤子線程。
  2. 避免耗時(shí)操作:鉤子線程中不要執(zhí)行長(zhǎng)時(shí)間任務(wù),否則會(huì)延遲 JVM 退出。
  3. 禁止異常拋出:鉤子線程中的異??赡軐?dǎo)致 JVM 無(wú)法正常退出。
  4. 注冊(cè)順序:按依賴關(guān)系注冊(cè)鉤子,先注冊(cè)簡(jiǎn)單任務(wù),后注冊(cè)復(fù)雜任務(wù)。
  5. 避免啟動(dòng)新線程:在鉤子中啟動(dòng)新線程可能導(dǎo)致 JVM 無(wú)法正常關(guān)閉。

開源框架中的關(guān)閉鉤子機(jī)制

1. Spring

在AbstractApplicationContext中,registerShutdownHook()方法注冊(cè)鉤子,用于關(guān)閉上下文:

public void registerShutdownHook() {
    if (this.shutdownHook == null) {
        this.shutdownHook = new Thread(() -> doClose());
        Runtime.getRuntime().addShutdownHook(this.shutdownHook);
    }
}

2. Tomcat

Tomcat 通過(guò)注冊(cè)鉤子確保服務(wù)關(guān)閉時(shí)釋放資源:

public void registerShutdownHook() {
    if (this.shutdownHook == null) {
        this.shutdownHook = new Thread(() -> {
            synchronized (startupShutdownMonitor) {
                doClose();
            }
        });
        Runtime.getRuntime().addShutdownHook(this.shutdownHook);
    }
}

關(guān)閉鉤子機(jī)制的原理

JVM 啟動(dòng)時(shí),主線程會(huì)創(chuàng)建一個(gè)關(guān)閉線程(Shutdown Thread),并將所有注冊(cè)的鉤子添加到其任務(wù)列表中。當(dāng) JVM 收到終止信號(hào)時(shí):

  1. 停止所有用戶線程。
  2. 啟動(dòng)關(guān)閉線程,按順序執(zhí)行鉤子任務(wù)。
  3. 等待所有鉤子執(zhí)行完畢或超時(shí)后退出。

鉤子的注冊(cè)與執(zhí)行

  • 注冊(cè):通過(guò)Runtime.getRuntime().addShutdownHook(Thread)將線程添加到ApplicationShutdownHooks的靜態(tài)列表中。
  • 執(zhí)行:關(guān)閉線程按順序同步執(zhí)行系統(tǒng)級(jí)鉤子,異步執(zhí)行應(yīng)用級(jí)鉤子,并等待所有線程完成。

關(guān)閉鉤子的觸發(fā)時(shí)機(jī)

  1. 主動(dòng)調(diào)用:通過(guò)Runtime.exit()或System.exit()觸發(fā)。
  2. 信號(hào)捕獲:JVM 注冊(cè)信號(hào)處理器(如INT、TERM),捕獲kill命令發(fā)送的信號(hào)后觸發(fā)。

示例代碼(捕獲信號(hào)):

public class SignalHandlerTest implements SignalHandler {
    public static void main(String[] args) {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> 
            System.out.println("關(guān)閉鉤子正在運(yùn)行...")));

        SignalHandler handler = new SignalHandlerTest();
        Signal.handle(new Signal("INT"), handler);  // 捕獲 Ctrl+C
        Signal.handle(new Signal("TERM"), handler); // 捕獲 kill 命令

        while (true) {
            System.out.println("主線程運(yùn)行中...");
            Thread.sleep(2000);
        }
    }

    @Override
    public void handle(Signal signal) {
        System.out.println("接收到信號(hào):" + signal.getName() + "-" + signal.getNumber());
        System.exit(0);
    }
}

輸出示例:

主線程運(yùn)行中...
主線程運(yùn)行中...
^C接收到信號(hào):INT-2
關(guān)閉鉤子正在運(yùn)行...

信號(hào)處理與守護(hù)線程

  • 信號(hào)不可捕獲的情況:KILL(9)和QUIT(3)無(wú)法被捕獲。
  • 守護(hù)線程:JVM 在所有用戶線程結(jié)束后自動(dòng)退出,守護(hù)線程(如 GC 線程)不會(huì)阻止 JVM 退出。

總結(jié)

Java 的關(guān)閉鉤子機(jī)制覆蓋了大部分退出場(chǎng)景,但以下情況例外:

  • 使用kill -9強(qiáng)制終止進(jìn)程時(shí),鉤子不會(huì)執(zhí)行。
  • 信號(hào)處理需調(diào)用System.exit()確保進(jìn)程退出。

通過(guò)合理使用關(guān)閉鉤子,可以實(shí)現(xiàn)資源釋放、服務(wù)優(yōu)雅關(guān)閉等關(guān)鍵功能。

責(zé)任編輯:武曉燕 來(lái)源: 程序猿技術(shù)充電站
相關(guān)推薦

2021-03-28 09:17:18

JVM場(chǎng)景鉤子函數(shù)

2025-08-04 06:00:00

Java并發(fā)編程開發(fā)

2021-01-19 10:35:49

JVM場(chǎng)景函數(shù)

2017-12-19 10:03:44

JavaLinux代碼

2024-12-24 08:03:56

2022-06-28 08:01:26

hook狀態(tài)管理state

2022-04-11 08:17:07

JVMJava進(jìn)程

2022-07-24 09:46:48

優(yōu)雅停機(jī)代碼

2020-12-01 12:44:44

PythonHook鉤子函數(shù)

2009-04-09 17:22:23

Linuxshutdown系統(tǒng)

2017-09-19 14:53:37

Java并發(fā)編程并發(fā)代碼設(shè)計(jì)

2021-04-20 08:00:31

Redisson關(guān)閉訂單支付系統(tǒng)

2023-12-20 10:04:45

線程池Java

2025-01-13 06:00:00

Go語(yǔ)言gRPC

2025-07-25 06:48:26

Java編程并發(fā)編程

2025-08-05 06:00:00

并發(fā)編程Java開發(fā)

2025-03-26 00:55:00

2025-03-20 06:48:55

性能優(yōu)化JDK

2025-06-18 08:10:00

Java并發(fā)編程開發(fā)

2025-02-17 00:00:25

Java并發(fā)編程
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)