首页 > 代码库 > JVM 关闭钩子

JVM 关闭钩子

1、功能

在jvm中添加关闭钩子(Runtime.getRuntime().addShutdownHook(shutdownHook);)后,当jvm关闭时会执行系统中已经设置的所有通过该方法添加的钩子,系统执行完这些钩子后,jvm才会关闭。所以这些钩子可以在jvm关闭的时候进行内存清理、对象销毁、关闭I/O资源等操作。

2、示例

示例1及输出:

技术分享
package cn.edu.buaa.jvmhook;/** * Runtime.getRuntime().addShutdownHook(shutdownHook); * <p> * 这个方法的意思就是在jvm中增加一个关闭的钩子,当jvm关闭的时候,会执行系统中已经设置的所有通过方法addShutdownHook添加的钩子, * 当系统执行完这些钩子后,jvm才会关闭。所以这些钩子可以在jvm关闭的时候进行内存清理、对象销毁等操作。 * </p> * 用途1:应用程序正常退出,在退出时执行特定的业务逻辑,或者关闭资源等操作。 */public class JVMHook1 {    public static void start() {        System.out.println("The JVM is started");        Runtime.getRuntime().addShutdownHook(new Thread() {            public void run() {                try {                    // do something                    System.out.println("The JVM Hook is execute");                } catch (Exception e) {                    e.printStackTrace();                }            }        });    }    public static void main(String[] args) {        start();        System.out.println("The Application is doing something");        try {            Thread.sleep(3000);        } catch (InterruptedException e) {            e.printStackTrace();        }        while (true) {            ;        }    }}The JVM is startedThe Application is doing something
View Code

示例2及输出:

技术分享
package cn.edu.buaa.jvmhook;/** * Runtime.getRuntime().addShutdownHook(shutdownHook); * <p> * 这个方法的意思就是在jvm中增加一个关闭的钩子,当jvm关闭的时候,会执行系统中已经设置的所有通过方法addShutdownHook添加的钩子, * 当系统执行完这些钩子后,jvm才会关闭。所以这些钩子可以在jvm关闭的时候进行内存清理、对象销毁等操作。 * </p> * 用途2:虚拟机非正常退出,比如用户按下ctrl+c、OutofMemory宕机、操作系统关闭等。在退出时执行必要的挽救措施。 */public class JVMHook2 {    public static void start() {        System.out.println("The JVM is started");        Runtime.getRuntime().addShutdownHook(new Thread() {            public void run() {                try {                    // do something                    System.out.println("The JVM Hook is execute");                } catch (Exception e) {                    e.printStackTrace();                }            }        });    }    public static void main(String[] args) {        start();        System.out.println("The Application is doing something");        byte[] b = new byte[-1];        System.out.println("The Application continues to do something");        try {            Thread.sleep(3000);        } catch (InterruptedException e) {            e.printStackTrace();        }    }}The JVM is startedThe Application is doing somethingException in thread "main" java.lang.NegativeArraySizeException    at cn.edu.buaa.jvmhook.JVMHook2.main(JVMHook2.java:31)The JVM Hook is execute
View Code

 

建议

同一个JVM最好只使用一个关闭钩子,而不是每个服务都使用一个不同的关闭钩子,使用多个关闭钩子可能会出现当前这个钩子所要依赖的服务可能已经被另外一个关闭钩子关闭了。为了避免这种情况,建议关闭操作在单个线程中串行执行,从而避免了再关闭操作之间出现竞态条件或者死锁等问题。

3、参考资料

http://www.cnblogs.com/langtianya/p/4300282.html

 

JVM 关闭钩子