Tomcat 关闭钩子

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

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

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

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

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

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

关闭钩子的生成: this

1.建立Thread的子类 url

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

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中的关闭钩子:

TomcatCatalina类中有一个钩子内部类的定义:

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方法用于关闭相关服务

Catalinastart方法中将钩子进行了注册:

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

咱们知道Tomcat启动时CatalinaStart方法会被调用,因此钩子在tomcat启动时就会被注册到虚拟

相关文章
相关标签/搜索