若有变化,就重新加载类,在这个过程中不会清空Session ,一般用在开发环境。
类似地,也由后台线程定时检测Web应用变化,但它会重新加载整个Web应用。这会清空Session,比热加载更干净、彻底,一般用在生产环境。
==============================================================================
Tomcat通过开启后台线程,使得各个层次的容器组件都有机会完成一些周期性任务。
实际开发往往也需要执行一些周期性任务,比如监控程序周期性拉取系统健康状态。
开启后台线程做周期性任务,最常见的就是线程池的ScheduledThreadPoolExecutor,没错,Tomcat就是通过它来开启后台线程:
backgroundProcessorFuture = Container.getService(this).getServer().getUtilityExecutor()
.scheduleWithFixedDelay(
// 要周期性执行的Runnable
new ContainerBackgroundProcessor(),
//第一次执行延迟多久
backgroundProcessorDelay,
// 之后每次执行间隔多久
backgroundProcessorDelay,
// 时间单位
TimeUnit.SECONDS);
任务类ContainerBackgroundProcessor是ContainerBase的内部类,ContainerBase是所有容器组件的基类。
protected class ContainerBackgroundProcessor implements Runnable {
@Override
public void run() {
// 入参"宿主类"实例
processChildren(ContainerBase.this);
}
protected void processChildren(Container container) {
try {
// 1. 调用当前容器的backgroundProcess
container.backgroundProcess();
// 2. 遍历所有子容器,递归调用processChildren
// 这样当前容器的子孙都会被处理
Container[] children = container.findChildren();
for (int i = 0; i < children.length; i++) {
// 容器基类有个变量叫做backgroundProcessorDelay
// 如果大于0,表明子容器有自己的后台线程
// 无需父容器来调用它的processChildren方法
if (children[i].getBackgroundProcessorDelay() <= 0) {
processChildren(children[i]);
}
}
} catch (Throwable t) { … }
processChildren把“宿主类”,即ContainerBase的类实例当成参数传给了run方法。
而在processChildren方法里,就做了两步:调用当前容器的backgroundProcess方法,以及递归调用子孙的backgroundProcess方法。请你注意backgroundProcess是Container接口中的方法,也就是说所有类型的容器都可以实现这个方法,在这个方法里完成需要周期性执行的任务。
这样只需在顶层容器Engine中启动一个后台线程,则该线程不但会执行Engine容器的周期性任务,还会执行所有子容器的周期性任务。
backgroundProcess方法
上述代码都是在基类ContainerBase实现,具体容器类需要做什么呢?
-
若有周期性任务,就实现backgroundProcess
-
若没有,则复用基类ContainerBase的方法
ContainerBase#backgroundProcess
public void backgroundProcess() {
// 1.执行容器中Cluster组件的周期性任务
Cluster cluster = getClusterInternal();
if (cluster != null) {
cluster.backgroundProcess();
}
// 2.执行容器中Realm组件的周期性任务
Realm realm = getRealmInternal();
if (realm != null) {
realm.backgroundProcess();
}
// 3.执行容器中Valve组件的周期性任务
Valve current = pipeline.getFirst();
while (current != null) {
current.backgroundProcess();
current = current.getNext();
}
// 4. 触发容器的"周期事件",Host容器的监听器HostConfig就靠它来调用
fireLifecycleEvent(Lifecycle.PERIODIC_EVENT, null);
}
不仅每个容器可以有周期性任务,每个容器中的其他通用组件,比如跟集群管理有关的Cluster组件、跟安全管理有关的Realm组件都可以有自己的周期性任务。
容器之间的链式调用是通过Pipeline-Valve机制实现的,容器中的Valve也可以有周期性任务,并且被ContainerBase统一处理。
在backgroundProcess方法的最后,还触发了容器的“周期事件”。“周期事件”是什么呢?
它跟生命周期事件一样,是一种扩展机制,可以这样理解:
又一段时间过去了,容器还活着,你想做点什么吗?
如果你想做点什么,就创建一个监听器来监听这个“周期事件”,事件到了我负责调用你的方法。
总之,有了ContainerBase中的后台线程和backgroundProcess方法,各种子容器和通用组件不需要各自弄一个后台线程来处理周期性任务。
========================================================================
有了ContainerBase的周期性任务处理“框架”,具体容器子类只需实现自己的周期性任务。
Tomcat的热加载,就实现在Context容器。
Context#backgroundProcess
StandardContext实现类中:
@Override
public void backgroundProcess() {
if (!getState().isAvailable())
return;
// 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();
}
super.backgroundProcess();
}
WebappLoader是如何实现热加载的呢?
关键是调用Context#reload方法:
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
4O-1715100476847)]
[外链图片转存中…(img-ZykpkvJR-1715100476848)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!