Tomcat 关闭钩子

使用JAVA的过程中,经常遇到程序启动时初始化一下资源,或生成一下临时文件,程序退出时要清除这些临时文件,或者程序退出时执行一下必要的其他操作。如果程序是通过我们提供的关闭/退出按钮正常退出的,一切还都好处理,但是如果用户直接关闭虚拟机运行的窗口,那一切就会变的比较复杂。

好在java提供了一种优雅的方式去解决这种问题。使得关闭的善后处理的代码能执行。java的关闭钩子能确保总是执行,无论用户如何终止应用程序。除非用户kill,这个是个死穴。

java而言,虚拟机会对以下几种操作进行关闭:

(1)系统调用System.exit()方法

(2)程序最后一个守护线程退出时,应用程序正常退出。

(3)用户强行中断程序运行,比如ctrl+c等其他方式中断java程序

关闭钩子的生成:

1.创建Thread的子类

2.实现run方法,应用程序关闭时会调用该方法,不需要调用start方法

3.在应用中实例化关闭钩子类

4.使用Runtime注册关闭钩子

下面是一个使用关闭钩子的例子:

public class ShutDownHook extends Thread{ 
    public static void main(String[] args){ 
        Runtime.getRuntime().addShutdownHook(new ShutDownHook()); 
       for(int i=0;i<10;i++){ 
        System.out.println("i="+i); 
       if(i==4){ 
           System.exit(0); 
       } 
         try { 
             Thread.sleep(1000); 
       } catch (InterruptedException e) { 
       // TODO Auto-generated catch block e.printStackTrace(); 
       } 
       }
     } 
    public void run(){
       System.out.println("hook shutdown!");
    } 
 }

使用Runtime.getRuntime().addShutdownHook方法注册钩子后,程序在非正常关闭是会执行钩子程序run方法中的相关代码。

Tomcat中的关闭钩子:

在Tomcat的Catalina类中有一个钩子内部类的定义:

protected class CatalinaShutdownHook extends Thread {
 public void run(){ 
  try {   
   if (getServer() != null) {  
    Catalina.this.stop(); 
   }  
   } catch (Throwable ex) {   
   log.error(sm.getString("catalina.shutdownHookFail"), ex);  
   } finally {  
   // If JULI is used, shut JULI down *after* the server shuts down  
   // so log messages aren‘t lost   
   LogManager logManager = LogManager.getLogManager();  
   if (logManager instanceof ClassLoaderLogManager) { 
   ((ClassLoaderLogManager) logManager).shutdown(); 
   } 
  } 
 }
}

钩子程序因为是Catalina的内部类,所以它可以调用Catalina类中的相关方法。其调用了stop方法用于关闭相关服务组件

在Catalina的start方法中将钩子进行了注册:

if (useShutdownHook) {if (shutdownHook == null) {shutdownHook = new CatalinaShutdownHook();}Runtime.getRuntime().addShutdownHook(shutdownHook);

我们知道Tomcat启动时Catalina的Start方法会被调用,所以钩子在tomcat启动时就会被注册到虚拟机

时间: 2024-12-25 08:13:48

Tomcat 关闭钩子的相关文章

Tomcat 学习进阶历程之关闭钩子

使用JAVA的过程中,经常遇到程序启动时初始化一下资源,或生成一下临时文件,程序退出时要清除这些临时文件,或者程序退出时执行一下必要的其他操作.如果程序是通过我们提供的关闭/退出按钮正常退出的,一切还都好处理,但是如果用户直接关闭虚拟机运行的窗口,那一切就会变的比较复杂. 好在java提供了一种优雅的方式去解决这种问题.使得关闭的善后处理的代码能执行.java的关闭钩子能确保总是执行,无论用户如何终止应用程序.除非用户kill,这个是个死穴. 对java而言,虚拟机会对以下几种操作进行关闭: (

tomat(16)关闭钩子

[0]REAMDE 0)本文部分文字描述转自:"how tomcat works",旨在学习"tomat(16)关闭钩子"的相关知识: 1)problem+solution: 1.1)problem:在很多实际环境中,当用户关闭应用程序时,并不会按照推荐的方法关闭应用程序,很有可能不做清理工作: 1.2)solution:java 为程序员提供了一种优雅的方法可以在在关闭过程中执行一些代码,以确保那些负责善后处理的代码可能能够执行: 2)在java中,虚拟机会对两类

关闭钩子

在很多实际应用环境中,当用户关了应用程序时,需要做一些善后清理工作,但问题是,用户有时并不会按照推荐的方法关闭应用程序,很有可能不做清理工作,例如在Tomcat的部署应用中,通过实例化一个Server对象来启动servlet容器,并调用其start方法,然后逐个调用组件的start方法,正常情况下,为了让Server对象能够关闭这些已经启动的组件,你应该向指定的端口发送关闭命令,如果你只是简单的突然退出,例如在应用程序过程中关闭控制台,可能会发生一些意想不到的事情. 幸运的是,java为程序员提

JVM 关闭钩子

1.功能 在jvm中添加关闭钩子(Runtime.getRuntime().addShutdownHook(shutdownHook);)后,当jvm关闭时会执行系统中已经设置的所有通过该方法添加的钩子,系统执行完这些钩子后,jvm才会关闭.所以这些钩子可以在jvm关闭的时候进行内存清理.对象销毁.关闭I/O资源等操作. 2.示例 示例1及输出: package cn.edu.buaa.jvmhook; /** * Runtime.getRuntime().addShutdownHook(shu

java的关闭钩子(Shutdown Hook)

Runtime.getRuntime().addShutdownHook(shutdownHook); 这个方法的含义说明: 这个方法的意思就是在jvm中增加一个关闭的钩子,当jvm关闭的时候,会执行系统中已经设置的所有通过方法addShutdownHook添加的钩子,当系统执行完这些钩子后,jvm才会关闭.所以这些钩子可以在jvm关闭的时候进行内存清理.对象销毁等操作. 用途 1应用程序正常退出,在退出时执行特定的业务逻辑,或者关闭资源等操作.   2虚拟机非正常退出,比如用户按下ctrl+c

Tomcat关闭过程(Tomcat源码解析四)

我们在Tomcat启动过程(Tomcat源代码阅读系列之三)一文中已经知道Tomcat启动以后,会启动6条线程,他们分别如下: "ajp-bio-8009-AsyncTimeout" daemon prio=5 tid=7f8738afe000 nid=0x115ad6000 waiting on condition [115ad5000] "ajp-bio-8009-Acceptor-0" daemon prio=5 tid=7f8738b05800 nid=0x

JAVA虚拟机关闭钩子(Shutdown Hook)

Java程序经常也会遇到进程挂掉的情况,一些状态没有正确的保存下来,这时候就需要在JVM关掉的时候执行一些清理现场的代码.JAVA中的ShutdownHook提供了比较好的方案. JDK提供了Java.Runtime.addShutdownHook(Thread hook)方法,可以注册一个JVM关闭的钩子,这个钩子可以在一下几种场景中被调用: 程序正常退出 使用System.exit() 终端使用Ctrl+C触发的中断 系统关闭 OutOfMemory宕机 使用Kill pid命令干掉进程(注

Tomcat 关闭时报错

最近tomcat走普通的关闭方式无法正常关闭,会报一些Error,用的是Tomcat7,据说是Tomcat7在关闭的时候加了一些检查线程泄漏内存泄露的东西 总结起来,在我项目中有这么几个原因会导致关闭不了: 1.使用了@Scheduled注解 最后查出来原因是因为在bean里面使用了  @Scheduled    注解,而Tomcat关闭的时候,spring并不能主动关闭这些Schedule,也就是说这注解有缺陷,慎用. 替代方法就是放弃spring的Scheduled注解方式使用quartz,

Tomcat关闭过程

我们在 Tomcat启动过程(Tomcat源代码阅读系列之三) 一文中已经知道Tomcat启动以后,会启动6条线程,他们分别如下: Tomcat threads 1 2 3 4 5 6 7 8 9 10 11 "ajp-bio-8009-AsyncTimeout" daemon prio=5 tid=7f8738afe000 nid=0x115ad6000 waiting on condition [115ad5000] "ajp-bio-8009-Acceptor-0&qu