Tomcat实现热部署、热加载原理解析

本文详细解释了热加载和热部署在JavaWeb开发中的实现原理,特别关注Tomcat如何通过后台线程管理和执行周期性任务,包括ContainerBase和ContainerBackgroundProcessor的作用。同时提到了如何通过现有项目和面试来检验技术学习成果。
摘要由CSDN通过智能技术生成
  • 热加载

  • 热部署

实现原理

===================================================================

跟类加载机制有关。

热加载


实现方式是Web容器启动一个后台线程,定期检测类文件变化。

若有变化,就重新加载类,在这个过程中不会清空Session ,一般用在开发环境。

热部署


类似地,也由后台线程定时检测Web应用变化,但它会重新加载整个Web应用。这会清空Session,比热加载更干净、彻底,一般用在生产环境。

Tomcat实现热加载、热部署

==============================================================================

Tomcat通过开启后台线程,使得各个层次的容器组件都有机会完成一些周期性任务。

实际开发往往也需要执行一些周期性任务,比如监控程序周期性拉取系统健康状态。

Tomcat后台线程


开启后台线程做周期性任务,最常见的就是线程池的ScheduledThreadPoolExecutor,没错,Tomcat就是通过它来开启后台线程:

backgroundProcessorFuture = Container.getService(this).getServer().getUtilityExecutor()

.scheduleWithFixedDelay(

// 要周期性执行的Runnable

new ContainerBackgroundProcessor(),

//第一次执行延迟多久

backgroundProcessorDelay,

// 之后每次执行间隔多久

backgroundProcessorDelay,

// 时间单位

TimeUnit.SECONDS);

任务类ContainerBackgroundProcessor是ContainerBase的内部类,ContainerBase是所有容器组件的基类。

ContainerBackgroundProcessor


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);

}

先自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以扫码领取!

img

如何快速更新自己的技术积累?

  • 在现有的项目里,深挖技术,比如用到netty可以把相关底层代码和要点都看起来。
  • 如果不知道目前的努力方向,就看自己的领导或公司里技术强的人在学什么。
  • 知道努力方向后不知道该怎么学,就到处去找相关资料然后练习。
  • 学习以后不知道有没有学成,则可以通过面试去检验。

我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

关资料然后练习。

  • 学习以后不知道有没有学成,则可以通过面试去检验。

我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

[外链图片转存中…(img-l7Jmt89q-1711461796816)]

[外链图片转存中…(img-W56Qw29W-1711461796817)]

需要更多Java资料的小伙伴可以帮忙点赞+关注,点击传送门,即可免费领取!

  • 12
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值