返回顶部
首页 > 资讯 > 精选 >ShutdownHook的原理是什么
  • 431
分享到

ShutdownHook的原理是什么

2023-06-15 14:06:35 431人浏览 安东尼
摘要

这篇文章主要讲解了“ShutdownHook的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“ShutdownHook的原理是什么”吧!ShutdownHook介绍在java程序中,

这篇文章主要讲解了“ShutdownHook的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“ShutdownHook的原理是什么”吧!

ShutdownHook介绍

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

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

有了ShutdownHook我们可以

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

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

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

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

  • 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) {             }         }     } }

用一副图总结如下:

ShutdownHook的原理是什么

ShutdownHook触发点

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

ShutdownHook的原理是什么

重点看Shutdown.exit 和 Shutdown.shutdown

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

这个方法可以看注释

翻译一下就是该方法会在最后一个非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() {             System.out.println("I'm shutdown hook ");         }     }); }

当main线程运行完了后,也能打印出I'm shutdown hook,反观Golang就做不到这一点

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

ShutdownHook的原理是什么

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

总结

综上,我们得出一些结论

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

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

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

感谢各位的阅读,以上就是“ShutdownHook的原理是什么”的内容了,经过本文的学习后,相信大家对ShutdownHook的原理是什么这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: ShutdownHook的原理是什么

本文链接: https://lsjlt.com/news/280495.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • ShutdownHook的原理是什么
    这篇文章主要讲解了“ShutdownHook的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“ShutdownHook的原理是什么”吧!ShutdownHook介绍在java程序中,...
    99+
    2023-06-15
  • Java ShutdownHook原理详解
    目录ShutdownHook介绍ShutdownHook原理ShutdownHook的数据结构与执行顺序ShutdownHook触发点Shutdown.exitShutdown.sh...
    99+
    2024-04-02
  • JSONP的原理是什么
    JSONP的原理是什么?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。什么是JSONP首先提一下JSON这个概念,JSON是一种轻量级的数据传输格式,被广泛应用于当前Web应用中...
    99+
    2023-06-14
  • Elasticsearch的原理是什么
    本篇内容主要讲解“Elasticsearch的原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Elasticsearch的原理是什么”吧!Lucene 和 ESLuceneLucene ...
    99+
    2023-06-16
  • CAS的原理是什么
    本篇内容介绍了“CAS的原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!在并发编程中我们都知道i++操作是非线程安全的,这是因为 i...
    99+
    2023-06-15
  • java.util.Random的原理是什么
    这篇文章给大家介绍java.util.Random的原理是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。java.util.Random可以产生int、long、float、double以及Goussian等类型的...
    99+
    2023-05-31
    java.util.random ava %util
  • Drupal的原理是什么
    小编今天带大家了解Drupal的原理是什么,文中知识点介绍的非常详细。觉得有帮助的朋友可以跟着小编一起浏览文章的内容,希望能够帮助更多想解决这个问题的朋友找到问题的答案,下面跟着小编一起深入学习“Drupal的原理是什么”的知识吧。Drup...
    99+
    2023-06-04
  • AOP的原理是什么
    这篇文章主要讲解了“AOP的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“AOP的原理是什么”吧!AOP的原理对哪些对象在什么位置拦截做什么 <=> <aop:b...
    99+
    2023-06-03
  • webservice的原理是什么
    Web服务的原理是通过使用HTTP协议进行通信,提供一组标准化的接口和方法,使不同的应用程序能够相互交互和通信。以下是Web服务的基...
    99+
    2024-02-29
    webservice
  • redis的原理是什么
    redis 是一款内存数据库,使用单线程架构和复杂的数据结构,实现了高性能和可扩展性。它支持数据持久化和复制,以确保数据安全性和可用性。 Redis 的原理 Redis 是一款开源的高...
    99+
    2024-04-20
    redis 数据丢失 键值对
  • JavaScript原型链的原理是什么
    这篇文章主要讲解了“JavaScript原型链的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JavaScript原型链的原理是什么”吧!解析原型...
    99+
    2024-04-02
  • Golang原生rpc的原理是什么
    这篇文章主要讲解了“Golang原生rpc的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Golang原生rpc的原理是什么”吧!创建rpc接口,需要几个条件方法的类型是可输出的方...
    99+
    2023-06-29
  • HTTP代理的原理是什么
    这篇文章主要讲解了“HTTP代理的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“HTTP代理的原理是什么”吧!这种情况在爬行动物的制作过程中经常发生。一开始,爬行动物正常工作,数据...
    99+
    2023-06-20
  • React Hooks的原理是什么
    这篇文章主要讲解了“React Hooks的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“React Hooks的原理是什么”吧!0x0...
    99+
    2024-04-02
  • div+css的原理是什么
    本篇内容主要讲解“div+css的原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“div+css的原理是什么”吧!一、思路分析   ...
    99+
    2024-04-02
  • nodejs​ npm的原理是什么
    这篇文章主要讲解了“nodejs npm的原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“nodejs npm的原理是什么”吧!npm的原理npm据...
    99+
    2024-04-02
  • Android ANR的原理是什么
    本篇内容介绍了“Android ANR的原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、ANR说明和原因1.1 简介A...
    99+
    2023-06-21
  • android leakcanary的原理是什么
    Android LeakCanary是一个用于检测内存泄漏的开源库。它的原理主要包括以下几个步骤:1. 监测对象的引用关系:Leak...
    99+
    2023-09-23
    android
  • spring scope的原理是什么
    Spring的Bean的作用域(scope)指定了一个Bean的实例是如何被创建和管理的。Spring框架提供了多种作用域,包括si...
    99+
    2023-08-31
    spring scope
  • spring session的原理是什么
    Spring Session是一种用于管理用户会话的框架,它通过将会话数据存储在外部存储介质中,而不是默认的内存中,来实现会话的持久...
    99+
    2023-09-21
    spring
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作