中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中Shutdown Hook怎么用

發布時間:2021-06-15 10:22:25 來源:億速云 閱讀:197 作者:小新 欄目:開發技術

小編給大家分享一下Java中Shutdown Hook怎么用,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

背景

如果想在 Java 進程退出時,包括正常和異常退出,做一些額外處理工作,例如資源清理,對象銷毀,內存數據持久化到磁盤,等待線程池處理完所有任務等等。特別是進程異常掛掉的情況,如果一些重要狀態沒及時保留下來,或線程池的任務沒被處理完,有可能會造成嚴重問題。那該怎么辦呢?

Java 中的 Shutdown Hook 提供了比較好的方案。我們可以通過 Java.Runtime.addShutdownHook(Thread hook) 方法向 JVM 注冊關閉鉤子,在 JVM 退出之前會自動調用執行鉤子方法,做一些結尾操作,從而讓進程平滑優雅的退出,保證了業務的完整性。

Shutdown Hook 介紹

其實,shutdown hook 就是一個簡單的已初始化但是未啟動的線程。當虛擬機開始關閉時,它將會調用所有已注冊的鉤子,這些鉤子執行是并發的,執行順序是不確定的。

在虛擬機關閉的過程中,還可以繼續注冊新的鉤子,或者撤銷已經注冊過的鉤子。不過有可能會拋出 IllegalStateException。注冊和注銷鉤子的方法定義如下:

public void addShutdownHook(Thread hook) {
 // 省略
}

public void removeShutdownHook(Thread hook) {
 // 省略
}

關閉鉤子被調用場景

關閉鉤子可以在以下幾種場景被調用:

  • 程序正常退出

  • 程序調用 System.exit() 退出

  • 終端使用 Ctrl+C 中斷程序

  • 程序拋出異常導致程序退出,例如 OOM,數組越界等異常

  • 系統事件,例如用戶注銷或關閉系統

  • 使用 Kill pid 命令殺掉進程,注意使用 kill -9 pid 強制殺掉不會觸發執行鉤子

驗證程序正常退出情況

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法...")));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
程序即將退出...
執行鉤子方法...

Process finished with exit code 0

驗證程序調用 System.exit() 退出情況

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法...")));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.exit(-1);
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
執行鉤子方法...

Process finished with exit code -1

驗證終端使用 Ctrl+C 中斷程序,在命令行窗口中運行程序,然后使用 Ctrl+C 中斷

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法...")));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.out.println("程序即將退出...");
    }
}

運行結果

D:\IdeaProjects\java-demo\java ShutdownHookDemo
程序開始啟動...
執行鉤子方法...

演示拋出異常導致程序異常退出

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法...")));
    }

    public static void main(String[] args) {
        System.out.println("程序開始啟動...");
        int a = 0;
        System.out.println(10 / a);
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
執行鉤子方法...
Exception in thread "main" java.lang.ArithmeticException: / by zero
 at com.chenpi.ShutdownHookDemo.main(ShutdownHookDemo.java:12)

Process finished with exit code 1

至于系統被關閉,或者使用 Kill pid 命令殺掉進程就不演示了,感興趣的可以自行驗證。

注意事項

可以向虛擬機注冊多個關閉鉤子,但是注意這些鉤子執行是并發的,執行順序是不確定的。

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法A...")));
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法B...")));
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法C...")));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
程序即將退出...
執行鉤子方法B...
執行鉤子方法C...
執行鉤子方法A...

向虛擬機注冊的鉤子方法需要盡快執行結束,盡量不要執行長時間的操作,例如 I/O 等可能被阻塞的操作,死鎖等,這樣就會導致程序短時間不能被關閉,甚至一直關閉不了。我們也可以引入超時機制強制退出鉤子,讓程序正常結束。

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            // 模擬長時間的操作
            try {
                Thread.sleep(1000000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.out.println("程序即將退出...");
    }
}

以上的鉤子執行時間比較長,最終會導致程序在等待很長時間之后才能被關閉。

如果 JVM 已經調用執行關閉鉤子的過程中,不允許注冊新的鉤子和注銷已經注冊的鉤子,否則會報 IllegalStateException 異常。通過源碼分析,JVM 調用鉤子的時候,即調用 ApplicationShutdownHooks#runHooks() 方法,會將所有鉤子從變量 hooks 取出,然后將此變量置為 null。

// 調用執行鉤子
static void runHooks() {
    Collection<Thread> threads;
    synchronized(ApplicationShutdownHooks.class) {
        threads = hooks.keySet();
        hooks = null;
    }

    for (Thread hook : threads) {
        hook.start();
    }
    for (Thread hook : threads) {
        try {
            hook.join();
        } catch (InterruptedException x) { }
    }
}

在注冊和注銷鉤子的方法中,首先會判斷 hooks 變量是否為 null,如果為 null 則拋出異常。

// 注冊鉤子
static synchronized void add(Thread hook) {
    if(hooks == null)
        throw new IllegalStateException("Shutdown in progress");

    if (hook.isAlive())
        throw new IllegalArgumentException("Hook already running");

    if (hooks.containsKey(hook))
        throw new IllegalArgumentException("Hook previously registered");

    hooks.put(hook, hook);
}
// 注銷鉤子
static synchronized boolean remove(Thread hook) {
    if(hooks == null)
        throw new IllegalStateException("Shutdown in progress");

    if (hook == null)
        throw new NullPointerException();

    return hooks.remove(hook) != null;
}

我們演示下這種情況

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            System.out.println("執行鉤子方法...");
            Runtime.getRuntime().addShutdownHook(new Thread(
                    () -> System.out.println("在JVM調用鉤子的過程中再新注冊鉤子,會報錯IllegalStateException")));
            // 在JVM調用鉤子的過程中注銷鉤子,會報錯IllegalStateException
            Runtime.getRuntime().removeShutdownHook(Thread.currentThread());
        }));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        Thread.sleep(2000);
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
程序即將退出...
執行鉤子方法...
Exception in thread "Thread-0" java.lang.IllegalStateException: Shutdown in progress
 at java.lang.ApplicationShutdownHooks.add(ApplicationShutdownHooks.java:66)
 at java.lang.Runtime.addShutdownHook(Runtime.java:211)
 at com.chenpi.ShutdownHookDemo.lambda$static$1(ShutdownHookDemo.java:8)
 at java.lang.Thread.run(Thread.java:748)

如果調用 Runtime.getRuntime().halt() 方法停止 JVM,那么虛擬機是不會調用鉤子的。

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> System.out.println("執行鉤子方法...")));
    }

    public static void main(String[] args) {
        System.out.println("程序開始啟動...");
        System.out.println("程序即將退出...");
        Runtime.getRuntime().halt(0);
    }
}

運行結果

程序開始啟動...
程序即將退出...

Process finished with exit code 0

如果要想終止執行中的鉤子方法,只能通過調用 Runtime.getRuntime().halt() 方法,強制讓程序退出。在Linux環境中使用 kill -9 pid 命令也是可以強制終止退出。

package com.chenpi;

public class ShutdownHookDemo {

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            System.out.println("開始執行鉤子方法...");
            Runtime.getRuntime().halt(-1);
            System.out.println("結束執行鉤子方法...");
        }));
    }

    public static void main(String[] args) {
        System.out.println("程序開始啟動...");
        System.out.println("程序即將退出...");
    }
}

運行結果

程序開始啟動...
程序即將退出...
開始執行鉤子方法...

Process finished with exit code -1

如果程序使用 Java Security Managers,使用 shutdown Hook 則需要安全權限 RuntimePermission(“shutdownHooks”),否則會導致 SecurityException。

實踐

例如,我們程序自定義了一個線程池,用來接收和處理任務。如果程序突然奔潰異常退出,這時線程池的所有任務有可能還未處理完成,如果不處理完程序就直接退出,可能會導致數據丟失,業務異常等重要問題。這時鉤子就派上用場了。

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class ShutdownHookDemo {
 // 線程池
    private static ExecutorService executorService = Executors.newFixedThreadPool(3);

    static {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            System.out.println("開始執行鉤子方法...");
            // 關閉線程池
            executorService.shutdown();
            try {
             // 等待60秒
                System.out.println(executorService.awaitTermination(60, TimeUnit.SECONDS));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("結束執行鉤子方法...");
        }));
    }

    public static void main(String[] args) throws InterruptedException {
        System.out.println("程序開始啟動...");
        // 向線程池添加10個任務
        for (int i = 0; i < 10; i++) {
            Thread.sleep(1000);
            final int finalI = i;
            executorService.execute(() -> {
                try {
                    Thread.sleep(4000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("Task " + finalI + " execute...");
            });
            System.out.println("Task " + finalI + " is in thread pool...");
        }
    }
}

在命令行窗口中運行程序,在10個任務都提交到線程池之后,任務都還未處理完成之前,使用 Ctrl+C 中斷程序,最終在虛擬機關閉之前,調用了關閉鉤子,關閉線程池,并且等待60秒讓所有任務執行完成。

Java中Shutdown Hook怎么用

Shutdown Hook 在 Spring 中的運用

Shutdown Hook 在 Spring 中是如何運用的呢。通過源碼分析,Springboot 項目啟動時會判斷 registerShutdownHook 的值是否為 true,默認是 true,如果為真則向虛擬機注冊關閉鉤子。

private void refreshContext(ConfigurableApplicationContext context) {
 refresh(context);
 if (this.registerShutdownHook) {
  try {
   context.registerShutdownHook();
  }
  catch (AccessControlException ex) {
   // Not allowed in some environments.
  }
 }
}

@Override
public void registerShutdownHook() {
 if (this.shutdownHook == null) {
  // No shutdown hook registered yet.
  this.shutdownHook = new Thread() {
   @Override
   public void run() {
    synchronized (startupShutdownMonitor) {
        // 鉤子方法
     doClose();
    }
   }
  };
  // 底層還是使用此方法注冊鉤子
  Runtime.getRuntime().addShutdownHook(this.shutdownHook);
 }
}

在關閉鉤子的方法 doClose 中,會做一些虛擬機關閉前處理工作,例如銷毀容器里所有單例 Bean,關閉 BeanFactory,發布關閉事件等等。

protected void doClose() {
 // Check whether an actual close attempt is necessary...
 if (this.active.get() && this.closed.compareAndSet(false, true)) {
  if (logger.isDebugEnabled()) {
   logger.debug("Closing " + this);
  }

  LiveBeansView.unregisterApplicationContext(this);

  try {
   // 發布Spring 應用上下文的關閉事件,讓監聽器在應用關閉之前做出響應處理
   publishEvent(new ContextClosedEvent(this));
  }
  catch (Throwable ex) {
   logger.warn("Exception thrown from ApplicationListener handling ContextClosedEvent", ex);
  }

  // Stop all Lifecycle beans, to avoid delays during individual destruction.
  if (this.lifecycleProcessor != null) {
   try {
       // 執行lifecycleProcessor的關閉方法
    this.lifecycleProcessor.onClose();
   }
   catch (Throwable ex) {
    logger.warn("Exception thrown from LifecycleProcessor on context close", ex);
   }
  }

  // 銷毀容器里所有單例Bean
  destroyBeans();

  // 關閉BeanFactory
  closeBeanFactory();

  // Let subclasses do some final clean-up if they wish...
  onClose();

  // Reset local application listeners to pre-refresh state.
  if (this.earlyApplicationListeners != null) {
   this.applicationListeners.clear();
   this.applicationListeners.addAll(this.earlyApplicationListeners);
  }

  // Switch to inactive.
  this.active.set(false);
 }
}

我們知道,我們可以定義 bean 并且實現 DisposableBean 接口,重寫 destroy 對象銷毀方法。destroy 方法就是在 Spring 注冊的關閉鉤子里被調用的。例如我們使用 Spring 框架的 ThreadPoolTaskExecutor 線程池類,它就實現了 DisposableBean 接口,重寫了 destroy 方法,從而在程序退出前,進行線程池銷毀工作。源碼如下:

@Override
public void destroy() {
 shutdown();
}

/**
 * Perform a shutdown on the underlying ExecutorService.
 * @see java.util.concurrent.ExecutorService#shutdown()
 * @see java.util.concurrent.ExecutorService#shutdownNow()
 */
public void shutdown() {
 if (logger.isInfoEnabled()) {
  logger.info("Shutting down ExecutorService" + (this.beanName != null ? " '" + this.beanName + "'" : ""));
 }
 if (this.executor != null) {
  if (this.waitForTasksToCompleteOnShutdown) {
   this.executor.shutdown();
  }
  else {
   for (Runnable remainingTask : this.executor.shutdownNow()) {
    cancelRemainingTask(remainingTask);
   }
  }
  awaitTerminationIfNecessary(this.executor);
 }
}

以上是“Java中Shutdown Hook怎么用”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

玉田县| 亚东县| 永年县| 陇川县| 思茅市| 永安市| 阿尔山市| 定州市| 边坝县| 乐陵市| 临城县| 大方县| 五台县| 松潘县| 南京市| 北碚区| 青海省| 沛县| 平顶山市| 贵阳市| 房产| 馆陶县| 旅游| 同心县| 塔城市| 乌拉特前旗| 丰原市| 安西县| 民权县| 南平市| 娄烦县| 夏河县| 登封市| 阿勒泰市| 南通市| 论坛| 五华县| 饶河县| 元阳县| 铁岭市| 芦山县|