在Tomcat中,通过Catalina的start()来启动服务器Server等组件;然后监听ShutDown命令来调用stop()关闭。
但是,如果程序没有走上面的流程,非正常关闭,那么stop()的逻辑就没法执行了。幸运的是,和Java的shutdown钩子一样,Tomcat也提供了自己的关闭钩子。
JVM的关闭钩子可以参考我的《JVM关闭钩子(2)—— 源码浅析》,主要是Shutdown和ApplicationShutdownHooks两个类。
Tomcat中的关闭钩子--CatalinaShutdownHook,作为一个内部类定义在Catalina中,它继承了Thread,并在run()中调用了Catalina.stop()。
protected class CatalinaShutdownHook extends Thread {
public void run() {
if (server != null)
Catalina.this.stop();
}
}
然后在Catalina.start()中,把CatalinaShutdownHook注册到Java的关闭钩子中。实际上Tomcat的关闭钩子就是注册到JVM的关闭钩子中。注册代码如下:
try {
if (useShutdownHook) {
if (shutdownHook == null)
shutdownHook = new CatalinaShutdownHook();
Runtime.getRuntime().addShutdownHook(shutdownHook);
}
} catch (Throwable t) { }
另外,在Catalina.stop()中有一段移除关闭钩子的代码,避免重复执行关闭操作。因为如果能正常关闭服务器的话,就不需要再执行一次关闭钩子了。移除代码如下:
try {
if (useShutdownHook)
Runtime.getRuntime().removeShutdownHook(shutdownHook);
} catch (Throwable t) { }
这样,当Tomcat异常关闭,没有执行Catalina.stop(),那么在JVM关闭的时候,就会触发关闭钩子,执行CatalinaShutdownHook.run(),来stop服务器,进行一些退出的清理工作。
——源代码取自Tomcat 5.0