Java ShutdownHook原理详解

作者:捉虫大师 时间:2023-11-10 21:30:36 

目录
  • ShutdownHook介绍

  • ShutdownHook原理

    • ShutdownHook的数据结构与执行顺序

    • ShutdownHook触发点

      • Shutdown.exit

      • Shutdown.shutdown

  • 总结

    ShutdownHook介绍

    在java程序中,很容易在进程结束时添加一个钩子,即ShutdownHook。通常在程序启动时加入以下代码即可


    Runtime.getRuntime().addShutdownHook(new Thread(){
       @Override
       public void run() {
           System.out.println("I'm shutdown hook...");
       }
    });

    有了ShutdownHook我们可以

    • 在进程结束时做一些善后工作,例如释放占用的资源,保存程序状态等

    • 为优雅(平滑)发布提供手段,在程序关闭前摘除流量

    不少java中间件或框架都使用了ShutdownHook的能力,如dubbo、spring等。

    spring中在application context被load时会注册一个ShutdownHook。 这个ShutdownHook会在进程退出前执行销毁bean,发出ContextClosedEvent等动作。 而dubbo在spring框架下正是监听了ContextClosedEvent,调用dubboBootstrap.stop()来实现清理现场和dubbo的优雅发布,spring的事件机制默认是同步的,所以能在publish事件时等待所有监听者执行完毕。

    ShutdownHook原理

    ShutdownHook的数据结构与执行顺序

    • 当我们添加一个ShutdownHook时,会调用ApplicationShutdownHooks.add(hook),往ApplicationShutdownHooks类下的静态变量private static IdentityHashMap<Thread, Thread> hooks添加一个hook,hook本身是一个thread对象

    • ApplicationShutdownHooks类初始化时会把hooks添加到Shutdown的hooks中去,而Shutdown的hooks是系统级的ShutdownHook,并且系统级的ShutdownHook由一个数组构成,只能添加10个

    • 系统级的ShutdownHook调用了thread类的run方法,所以系统级的ShutdownHook是同步有序执行的


    private static void runHooks() {
       for (int i=0; i < MAX_SYSTEM_HOOKS; i++) {
           try {
               Runnable hook;
               synchronized (lock) {
                   // acquire the lock to make sure the hook registered during
                   // shutdown is visible here.
                   currentRunningHook = i;
                   hook = hooks[i];
               }
               if (hook != null) hook.run();
           } catch(Throwable t) {
               if (t instanceof ThreadDeath) {
                   ThreadDeath td = (ThreadDeath)t;
                   throw td;
               }
           }
       }
    }
    • 系统级的ShutdownHook的add方法是包可见,即我们不能直接调用它

    • ApplicationShutdownHooks位于下标1处,且应用级的hooks,执行时调用的是thread类的start方法,所以应用级的ShutdownHook是异步执行的,但会等所有hook执行完毕才会退出。


    static void runHooks() {
       Collection<Thread> threads;
       synchronized(ApplicationShutdownHooks.class) {
           threads = hooks.keySet();
           hooks = null;
       }
    for (Thread hook : threads) {
           hook.start();
       }
       for (Thread hook : threads) {
           while (true) {
               try {
                   hook.join();
                   break;
               } catch (InterruptedException ignored) {
               }
           }
       }
    }

    用一副图总结如下:

    Java ShutdownHook原理详解

    ShutdownHook触发点

    从Shutdown的runHooks顺藤摸瓜,我们得出以下这个调用路径

    Shutdown.exit

    跟进Shutdown.exit的调用方,发现有 Runtime.exit 和 Terminator.setup

    • Runtime.exit 是代码中主动结束进程的接口

    • Terminator.setup 被 initializeSystemClass 调用,当第一个线程被初始化的时候被触发,触发后注册了一个信号监控函数,捕获kill发出的信号,调用Shutdown.exit结束进程

    这样覆盖了代码中主动结束进程和被kill杀死进程的场景。

    主动结束进程不必介绍,这里说一下信号捕获。在java中我们可以写出如下代码来捕获kill信号,只需要实现SignalHandler接口以及handle方法,程序入口处注册要监听的相应信号即可,当然不是每个信号都能捕获处理。


    public class SignalHandlerTest implements SignalHandler {
    public static void main(String[] args) {
    Runtime.getRuntime().addShutdownHook(new Thread() {
               @Override
               public void run() {
                   System.out.println("I'm shutdown hook ");
               }
           });
    SignalHandler sh = new SignalHandlerTest();
           Signal.handle(new Signal("HUP"), sh);
           Signal.handle(new Signal("INT"), sh);
           //Signal.handle(new Signal("QUIT"), sh);// 该信号不能捕获
           Signal.handle(new Signal("ABRT"), sh);
           //Signal.handle(new Signal("KILL"), sh);// 该信号不能捕获
           Signal.handle(new Signal("ALRM"), sh);
           Signal.handle(new Signal("TERM"), sh);
    while (true) {
               System.out.println("main running");
               try {
                   Thread.sleep(2000L);
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       }
    @Override
       public void handle(Signal signal) {
           System.out.println("receive signal " + signal.getName() + "-" + signal.getNumber());
           System.exit(0);
       }
    }

    要注意的是通常来说,我们捕获信号,做了一些个性化的处理后需要主动调用System.exit,否则进程就不会退出了,这时只能使用kill -9来强制杀死进程了。

    而且每次信号的捕获是在不同的线程中,所以他们之间的执行是异步的。

    Shutdown.shutdown

    这个方法可以看注释


    /* Invoked by the JNI DestroyJavaVM procedure when the last non-daemon
     * thread has finished.  Unlike the exit method, this method does not
     * actually halt the VM.
     */

    翻译一下就是该方法会在最后一个非daemon线程(非守护线程)结束时被JNI的DestroyJavaVM方法调用。

    java中有两类线程,用户线程和守护线程,守护线程是服务于用户线程,如GC线程,JVM判断是否结束的标志就是是否还有用户线程在工作。 当最后一个用户线程结束时,就会调用 Shutdown.shutdown。这是JVM这类虚拟机语言特有的"权利",倘若是golang这类编译成可执行的二进制文件时,当全部用户线程结束时是不会执行ShutdownHook的。

    举个例子,当java进程正常退出时,没有在代码中主动结束进程,也没有kill,就像这样


    public static void main(String[] args) {
    Runtime.getRuntime().addShutdownHook(new Thread() {
           @Override
           public void run() {
               super.run();
               System.out.println("I'm shutdown hook ");
           }
       });
    }

    当main线程运行完了后,也能打印出I'm shutdown hook,反观golang就做不到这一点(如果可以做到,可以私信告诉我,我是个golang新手)

    通过如上两个调用的分析,我们概括出如下结论:

    Java ShutdownHook原理详解

    我们能看出java的ShutdownHook其实覆盖的非常全面了,只有一处无法覆盖,即当我们杀死进程时使用了kill -9时,由于程序无法捕获处理,进程被直接杀死,所以无法执行ShutdownHook。

    总结

    综上,我们得出一些结论

    • 重写捕获信号需要注意主动退出进程,否则进程可能永远不会退出,捕获信号的执行是异步的

    • 用户级的ShutdownHook是绑定在系统级的ShutdownHook之上,且用户级是异步执行,系统级是同步顺序执行,用户级处于系统级执行顺序的第二位

    • ShutdownHook 覆盖的面比较广,不论是手动调用接口退出进程,还是捕获信号退出进程,抑或是用户线程执行完毕退出,都会执行ShutdownHook,唯一不会执行的就是kill -9

    来源:https://juejin.cn/post/6954972766994956318

    标签:Java,ShutdownHook
    0
    投稿

    猜你喜欢

  • Mapper类中存在名称相同的方法重载报错问题

    2023-04-04 02:44:39
  • Java核心编程之文件随机读写类RandomAccessFile详解

    2023-11-28 17:40:05
  • springmvc 分页查询的简单实现示例代码

    2022-01-09 11:08:22
  • 深入理解JAVA基础类库中对象Object类

    2023-07-26 12:24:10
  • Java计算两个程序运行时间的实例

    2022-01-30 14:49:06
  • Android 多媒体播放API简单实例

    2022-12-12 18:53:24
  • C#遍历List并删除某个元素的方法

    2023-03-22 10:22:08
  • java 按行读取文件并输出到控制台的方法

    2022-10-28 22:48:20
  • Android开发之完成登陆界面的数据保存回显操作实例

    2022-07-05 19:13:40
  • Java Web项目部署在Tomcat运行出错与解决方法示例

    2023-07-18 02:50:47
  • JAVA实现红包分发的示例代码

    2022-10-08 06:18:15
  • 详解Android中提示对话框(ProgressDialog和DatePickerDialog和TimePickerDialog&PopupWindow)

    2023-05-10 19:27:43
  • Android开发之文件操作模式深入理解

    2023-10-14 02:48:12
  • ConcurrentMap.putIfAbsent(key,value)用法实例

    2023-03-14 06:45:43
  • Android实战打飞机游戏之怪物(敌机)类的实现(4)

    2021-07-26 09:13:41
  • 基于java的opencv开发过程详解

    2022-03-31 20:02:59
  • Struts 2中的constant配置详解

    2023-11-10 08:18:18
  • Android 7.0新特性详解

    2022-10-10 07:11:56
  • Java 7大常见排序方法实例详解

    2022-01-09 05:16:46
  • Android手势密码实现实例代码

    2023-04-13 20:17:51
  • asp之家 软件编程 m.aspxhome.com