要在运行的过程中升级Web应用,如果你不想重启系统,实现的方式有两种:热加载和热部署。
那如何实现热部署和热加载呢?它们跟类加载机制有关,具体来说就是:
- 热加载的实现方式是Web容器启动一个后台线程,定期检测类文件的变化,如果有变化,就重新加载类,在这个过程中不会清空Session ,一般用在开发环境。
- 热部署原理类似,也是由后台线程定时检测Web应用的变化,但它会重新加载整个Web应用。这种方式会清空Session,比热加载更加干净、彻底,一般用在生产环境。
Tomcat热加载
Tomcat的热加载,就是在Context容器中实现的。Context容器的backgroundProcess方法是这样实现的:
public void backgroundProcess() {
//WebappLoader周期性的检查WEB-INF/classes和WEB-INF/lib⽬录下的类⽂件
Loader loader = getLoader();
if (loader != null) {
loader.backgroundProcess();
}
//Session管理器周期性的检查是否有过期的Session
Manager manager = getManager();
if (manager != null) {
manager.backgroundProcess();
}
//周期性的检查静态资源是否有变化
WebResourceRoot resources = getResources();
if (resources != null) {
resources.backgroundProcess();
}
//调⽤⽗类ContainerBase的backgroundProcess⽅法
super.backgroundProcess();
}
从上面的代码我们看到Context容器通过WebappLoader来检查类文件是否有更新,通过Session管理器来检查是否有Session过期,并且通过资源管理器来检查静态资源是否有更新,最后还调用了父类ContainerBase的backgroundProcess方法。
这里我们要重点关注,WebappLoader是如何实现热加载的,它主要是调用了Context容器的reload方法,而Context的reload方法比较复杂,总结起来,主要完成了下面这些任务:
- 停止和销毁Context容器及其所有子容器,子容器其实就是Wrapper,也就是说Wrapper里面Servlet实例也被销毁了。
- 停止和销毁Context容器关联的Listener和Filter。
- 停止和销毁Context下的Pipeline和各种Valve。
- 停止和销毁Context的类加载器,以及类加载器加载的类文件资源。
- 启动Context容器,在这个过程中会重新创建前面四步被销毁的资源。
在这个过程中,类加载器发挥着关键作用。一个Context容器对应一个类加载器,类加载器在销毁的过程中会把它加载的所有类也全部销毁。Context容器在启动过程中,会创建一个新的类加载器来加载新的类文件。
在Context的reload方法里,并没有调用Session管理器的distroy方法,也就是说这个Context关联的Session是没有销毁的。你还需要注意的是,Tomcat的热加载默认是关闭的,你需要在conf目录下的Context.xml文件中设置reloadable参数来开启这个功能,像下面这样:
<Context reloadable="true"/>
Tomcat热部署
热部署跟热加载的本质区别是,热部署会重新部署Web应用,原来的Context对象会整个被销毁掉,因此这个Context所关联的一切资源都会被销毁,包括Session。
那么Tomcat热部署又是由哪个容器来实现的呢?答案是由Host,因为热部署过程中Context容器被销毁了,那么这个重担就落在Host身上了,因为它是Context的父容器。
跟Context不一样,Host容器并没有在backgroundProcess方法中实现周期性检测的任务,而是通过监听器HostConfig来实现的。
其实HostConfig会检查webapps目录下的所有Web应用:
- 如果原来Web应用目录被删掉了,就把相应Context容器整个销毁掉。
- 是否有新的Web应用目录放进来了,或者有新的WAR包放进来了,就部署相应的Web应用。
总结
Tomcat的热加载和热部署,它们的目的都是在不重启Tomcat的情况下实现Web应用的更新。
热加载的粒度比较小,主要是针对类文件的更新,通过创建新的类加载器来实现重新加载。而热部署是针对整个Web应用的,Tomcat会将原来的Context对象整个销毁掉,再重新创建Context容器对象。
热加载和热部署的实现都离不开后台线程的周期性检查,Tomcat在基类ContainerBase中统一实现了后台线程的处理逻辑,并在顶层容器Engine启动后台线程,这样子容器组件甚至各种通用组件都不需要自己去创建后台线程,这样的设计显得优雅整洁。
网友评论