public class Main extends Thread { @Override public void run() { for(int i = 0; i < 20; i++){ System.out.println(i); try { TimeUnit.SECONDS.sleep(3L); } catch (InterruptedException e) { // ignore } } } public static void main(String[] args) { new Main().start(); // thread /* shutdown hook */ Runtime.getRuntime().addShutdownHook( new Thread(){ public void run() { System.out.println("this is shutdown hook"); /* while(true){ // 发生死锁 } */ } } ); //Runtime.getRuntime().exit(0); // 等同于 System.exit(n); 会调用hook Runtime.getRuntime().halt(0); // 不调用 hook System.err.println("the main exit(0)"); } }
如程序中写的那样,如果hook发生死锁,那么Runtime.getRuntime().exit(0)或者System.exit(0)将永远不会被执行.这里介绍下ShutdownHook,Java 虚拟机会为了响应以下两类事件而关闭:
- 1.程序正常退出,这发生在最后的非守护线程退出时,或者在调用
exit
(等同于System.exit
)方法时。- 2.为响应用户中断而终止虚拟机,如键入 ^C;或发生系统事件,比如用户注销或系统关闭。
关闭hook只是一个已初始化但尚未启动的线程。虚拟机开始启用其关闭序列时,它会以某种未指定的顺序启动所有已注册的关闭钩子,并让它们同时运行。运行完所有的钩子后,如果已启用退出终结,那么虚拟机接着会运行所有未调用的终结方法。最后,虚拟机会暂停。注意,关闭序列期间会继续运行守护线程,如果通过调用
exit
方法来发起关闭序列,那么也会继续运行非守护线程。一旦开始了关闭序列,则只能通过调用
halt
方法来停止这个序列,此方法可强行终止虚拟机。