Tomcat源码解析——类加载机制

        一、类加载器的创建

        在之前的Tomcat启动源码中,简单的介绍了Tomcat的四种类加载器,再复习一遍。

类加载器

作用父加载器
commonLoader(共同类加载器)加载$CATALINA_HOME/lib下的类加载器应用类加载器
catalinaLoader(容器类加载器)加载Tomcat应用服务器的类加载器,可以理解为加载Tomcat源码中的类共同类加载器
sharedLoader(共享类加载器)加载应用类加载器的共享的类加载器,例如相同版本的mysql驱动等共同类加载器
webappLoader(Web应用类加载)加载Web应用下的类类加载,每个web应用之间是相互隔离的共享类加载器

        类加载器的结构层次:

                

         commonLoader、catalinaLoader、sharedLoader可以在tomcat下的conf/catalina.properties文件中修改。

        在Tomcat的启动中,一开始就创建了commonLoader、catalinaLoader、sharedLoader类加载器并且加载对应设置的资源。

Bootstrap:private void initClassLoaders() {try {//读取catalina.properties下的common.loader资源加载//commonLoader的父类加载器是应用类加载器(不设置都默认为应用类加载器)commonLoader = createClassLoader("common", null);if( commonLoader == null ) {commonLoader=this.getClass().getClassLoader();}//读取catalina.properties下的server.loader资源加载//catalinaLoader的父类加载器是commonLoadercatalinaLoader = createClassLoader("server", commonLoader);//读取catalina.properties下的shard.loader资源加载//sharedLoader的父类加载器是commonLoadersharedLoader = createClassLoader("shared", commonLoader);} catch (Throwable t) {handleThrowable(t);log.error("Class loader creation threw exception", t);System.exit(1);}}private ClassLoader createClassLoader(String name, ClassLoader parent)throws Exception {//读取catalina.properties中对应的key值String value = CatalinaProperties.getProperty(name + ".loader");//如果没有设值对应的属性值,则返回父类加载器if ((value == null) || (value.equals("")))return parent;//替换掉Tomact的表达式变量值,如catalina.home、catalina.base等为具体的路径value = replace(value);List<Repository> repositories = new ArrayList<Repository>();StringTokenizer tokenizer = new StringTokenizer(value, ",");while (tokenizer.hasMoreElements()) {String repository = tokenizer.nextToken().trim();if (repository.length() == 0) {continue;}try {//创建一个URL@SuppressWarnings("unused")URL url = new URL(repository);repositories.add(new Repository(repository, RepositoryType.URL));continue;} catch (MalformedURLException e) {// Ignore}if (repository.endsWith("*.jar")) {repository = repository.substring(0, repository.length() - "*.jar".length());repositories.add(new Repository(repository, RepositoryType.GLOB));} else if (repository.endsWith(".jar")) {repositories.add(new Repository(repository, RepositoryType.JAR));} else {repositories.add(new Repository(repository, RepositoryType.DIR));}}//创建一个URL的类加载器,并把要加载的路径传递过去return ClassLoaderFactory.createClassLoader(repositories, parent);}

        严格来说,commonLoader和sharedLoader都算是可以共享的类加载器,只是作用域不同,所以在catalina.properties中设置了common.loader和shared.loader加载路径时,则在应用的maven中需要排除掉这些共享的依赖(不排除的话,会在web类加载器中重复加载,common.loader和server.loader设置就无意义)。

        commonLoader是catalinaLoader和sharedLoader的父类加载器,如果server.loader和shared.loader没有设值时,那么这3个类加载器的值都是一样的。

        Web应用类加载器的创建是在Host容器启动之后,Host容器会发送事件到HostConfig中,然后启动Host下的应用。

HostConfig:public void lifecycleEvent(LifecycleEvent event) {try {//Host的生命周期事件host = (Host) event.getLifecycle();if (host instanceof StandardHost) {setCopyXML(((StandardHost) host).isCopyXML());setDeployXML(((StandardHost) host).isDeployXML());setUnpackWARs(((StandardHost) host).isUnpackWARs());setContextClass(((StandardHost) host).getContextClass());}} catch (ClassCastException e) {log.error(sm.getString("hostConfig.cce", event.getLifecycle()), e);return;}if (event.getType().equals(Lifecycle.PERIODIC_EVENT)) {check();} else if (event.getType().equals(Lifecycle.START_EVENT)) {//Host启动之后,部署启动Host下的应用start();} else if (event.getType().equals(Lifecycle.STOP_EVENT)) {stop();}}public void start() {//...省略if (host.getDeployOnStartup())//部署应用,即启动应用deployApps();}protected void deployApps() {File appBase = appBase();File configBase = configBase();String[] filteredAppPaths = filterAppPaths(appBase.list());deployDescriptors(configBase, configBase.list());//热部署War包deployWARs(appBase, filteredAppPaths);//部署扩展文件夹,即War包的解压deployDirectories(appBase, filteredAppPaths);}

        不管是如何部署,最终都会调用Context的start方法启动应用,在Context启动时,会创建对应的Web应用类加载器进行绑定,即一个Context对应一个Web应用类加载器,从而实现应用之间Jar包的隔离。

        

WebappLoader:private String loaderClass ="org.apache.catalina.loader.WebappClassLoader";protected void startInternal() throws LifecycleException {//...省略try {//创建应用对应的Web应用类加载器classLoader = createClassLoader();classLoader.setResources(container.getResources());classLoader.setDelegate(this.delegate);classLoader.setSearchExternalFirst(searchExternalFirst);if (container instanceof StandardContext) {classLoader.setAntiJARLocking(((StandardContext) container).getAntiJARLocking());classLoader.setClearReferencesStatic(((StandardContext) container).getClearReferencesStatic());classLoader.setClearReferencesStopThreads(((StandardContext) container).getClearReferencesStopThreads());classLoader.setClearReferencesStopTimerThreads(((StandardContext) container).getClearReferencesStopTimerThreads());classLoader.setClearReferencesHttpClientKeepAliveThread(((StandardContext) container).getClearReferencesHttpClientKeepAliveThread());}//...省略}private WebappClassLoaderBase createClassLoader()throws Exception {Class<?> clazz = Class.forName(loaderClass);WebappClassLoaderBase classLoader = null;if (parentClassLoader == null) {parentClassLoader = container.getParentClassLoader();}Class<?>[] argTypes = { ClassLoader.class };Object[] args = { parentClassLoader };Constructor<?> constr = clazz.getConstructor(argTypes);classLoader = (WebappClassLoaderBase) constr.newInstance(args);return classLoader;}

        可以看到,默认创建的Web应用类加载器是WebappClassLoader,同时会设置父类加载。

        通过打断点的方式,可以看到Web应用类加载器的父类加载器是共享类加载器。

        二、类加载器的使用

        在上文中,Tomcat的四大类加载器已经创建完毕,那么Tomcat是如何实现应用之间的隔离?

        在上一篇Tomcat的请求流程中,我们知道当一个请求到达时,会经过容器通过Valve传递。

StandardHostValve:public final void invoke(Request request, Response response)throws IOException, ServletException {//...省略//为当前线程设置应用的类加载器,从而实现应用之间隔离Thread.currentThread().setContextClassLoader(context.getLoader().getClassLoader());//...省略//传递到下一个容器的Valve中context.getPipeline().getFirst().invoke(request, response)}

        当请求到达HostValve中,会为当前线程设置类加载器为对应的Web应用类加载器,那么JVM在后面加载类时,会使用该类加载器。(JVM的机制)

        WebappClassLoader是默认的Web应用类加载器,重写了loadClass方法。

WebappClassLoader:public Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {synchronized (getClassLoadingLockInternal(name)) {Class<?> clazz = null;//先查看本地缓存是否加载了该类clazz = findLoadedClass0(name);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}//查询当前的类加载器是否已经加载了clazz = findLoadedClass(name);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}//尝试用扩展类加载器加载该类(依旧遵循双亲委派机制),防止Java的核心类被破坏try {clazz = j2seClassLoader.loadClass(name);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}} catch (ClassNotFoundException e) {}//检查访问权限if (securityManager != null) {int i = name.lastIndexOf('.');if (i >= 0) {try {securityManager.checkPackageAccess(name.substring(0,i));} catch (SecurityException se) {String error = "Security Violation, attempt to use " +"Restricted Class: " + name;throw new ClassNotFoundException(error, se);}}}boolean delegateLoad = delegate || filter(name);//当delegate为true时,会继续使用双亲委派加载方式,不过默认都是falseif (delegateLoad) {if (log.isDebugEnabled())log.debug("  Delegating to parent classloader1 " + parent);try {clazz = Class.forName(name, false, parent);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}} catch (ClassNotFoundException e) {}}try {//查找加载应用目录的class(/WEB-INF/classes和/WEB-INF/lib,既war包中的所有类)clazz = findClass(name);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}} catch (ClassNotFoundException e) {}//如果本地的class还没加载没有找到,则使用父加载器加载继续尝试(此处会走双亲委派)if (!delegateLoad) {try {//使用父加载器加载,此处会走双亲委派的机制clazz = Class.forName(name, false, parent);if (clazz != null) {if (resolve)resolveClass(clazz);return (clazz);}} catch (ClassNotFoundException e) {}}}throw new ClassNotFoundException(name);}

        应用类加载器的加载流程:

                1.查询缓存是否已经加载,加载不成功则进行下一步

                2.使用扩展类加载器加载(防止Java核心类被破坏),加载不成功则进行下一步

                3.使用Web应用类加载器加载(加载/WEB-INF/classes和/WEB-INF/lib),加载不成功则进行下一步

                4.交给父类加载器走双亲委派加载,加载路径则为:共享类加载器——>共同类加载器——>应用类加载器——>扩展类加载器——>系统类加载器      

        每个应用都使用自己的Web应用类加载器加载/WEB-INF/classes和/WEB-INF/lib,从而实现了Tomcat之间的应用隔离。

三、为什么要打破双亲委派机制?

        如果不打破双亲委派机制,能实现应用之间的隔离吗?

        答案是可以的,如果每个应用都创建一个自己的类加载器,走双亲委派加载时,最终还是在该类加载器实现最终的加载。

        既然如此,为什么Tomcat要打破双亲委派机制呢?

        因为Tomcat需要节约资源,如果走了双亲委派机制,那么一些共同的类库将无法实现共享,每个应用的类加载器都需要把所有的类库全部加载到自己的类加载器中,会浪费很多的内存资源,打破双亲委派机制,不仅可以让共同使用的类库实现共享,还能实现应用之间的隔离,不造成内存资源的浪费。

        

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/1057.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Games101-光线追踪(辐射度量学、渲染方程与全局光照)

Basic radiometry (辐射度量学) 光的强度假定l为10&#xff0c;但是10是什么。 Whitted-Style中间了很多不同简化&#xff0c;如能看到高光&#xff0c;表示做了布林冯着色&#xff0c;意味着一个光线打进来后会被反射到一定的区域里&#xff0c;而不是沿着完美的镜像方向&…

html select 支持内容过滤列表 -bootstrap实现

实现使用bootstrap-select插件 http://silviomoreto.github.io/bootstrap-select <!DOCTYPE html> <html> <meta charset"UTF-8"> <head><title>jQuery bootstrap-select可搜索多选下拉列表插件-www.daimajiayuan.com</title>&…

【Git】常用命令速查

目录 一、创建版本 二、修改和提交 三、查看提交历史 四、撤销 五、分支与标签 六、合并与衍合 七、远程操作 一、创建版本 命令简要说明注意事项git clone <url>克隆远程版本库 二、修改和提交 命令简要说明注意事项 三、查看提交历史 命令简要说明注意事项 …

Ribbon 添加快速访问区域

添加快速访问区域挺简单的&#xff0c;实例如下所示&#xff1a; void QtRightFuncDemo::createQuickAccessBar() { RibbonQuickAccessBar* quickAccessBar ribbonBar()->quickAccessBar(); QAction* action quickAccessBar->actionCustomizeButton(); act…

最邻近插值和线性插值

最邻近插值 在图像分割任务中&#xff1a;原图的缩放一般采用双线性插值&#xff0c;用于上采样或下采样&#xff1b;而标注图像的缩放有特定的规则&#xff0c;需使用最临近插值&#xff0c;不用于上采样或下采样。 自定义函数 这个是通过输入原始图像和一个缩放因子来对图像…

[NISACTF 2022]huaji?

注意要加--run-asroot

基于ThinkPHP框架开发的的站长在线工具箱网站PHP源码(可以作为流量站)

这是一套基于ThinkPHP框架开发的站长在线工具箱网站PHP源码&#xff0c;包含了多种在线工具&#xff0c;可以作为流量站使用。 项 目 地 址 &#xff1a; runruncode.com/php/19742.html 部署教程&#xff1a; 环境要求&#xff1a; - PHP版本需要大于等于7.2.5 - MySQL版…

2024年适用于 Android 的最佳免费数据恢复应用程序

无论是系统崩溃、软件升级、病毒攻击还是任何其他故障&#xff0c;这些软件问题都可能导致手机上的数据丢失。可以使用免费的数据恢复应用程序修复数据故障并检索丢失或删除的文件。 数据恢复应用程序旨在从另一个存储设备中检索丢失或无法访问的数据。这些工具扫描 UFS 并尝试…

视频素材库哪里最好?8个视频素材免费商用

在视频创作的世界里&#xff0c;寻找那些能够完美匹配你的想法并加以实现的视频素材是一项既令人兴奋又充满挑战的任务。无论你的目标是提升视频质量、增强视觉效果&#xff0c;还是简单地想要让你的作品更加出色&#xff0c;这里有一系列全球精选的视频素材网站&#xff0c;它…

Unity Editor编辑器扩展之创建脚本

前言 既然你看到这篇文章了&#xff0c;你是否也有需要使用代码创建脚本的需求&#xff1f;使用编辑器扩展工具根据不同的表格或者新增的内容去创建你想要的脚本。如果不使用工具&#xff0c;那么你只能不断去修改某个脚本&#xff0c;这项工作既繁琐也浪费时间。这个时候作为程…

PyTorch and Stable Diffusion on FreeBSD

Stable Diffusion在图像生成领域具有广泛的应用和显著的优势。它利用深度学习和扩散模型的原理&#xff0c;能够从随机噪声中生成高质量的图像。 官网&#xff1a;GitHub - verm/freebsd-stable-diffusion: Stable Diffusion on FreeBSD with CUDA support FreeBSD下难度主要…

【Linux 杂货铺】进程间通信

1.进程为什么要通信呢&#xff1f; ①&#x1f34e; 为了进程之间更好的协同工作&#xff0c;举个例子&#xff0c;在学校&#xff0c;学院的管理人员给教师安排课程的时候&#xff0c;必须事先知道该教师平常的上课情况&#xff0c;不然会将教师的课程安排到一起造成麻烦&…

偏微分方程算法之二维初边值问题(紧交替方向隐格式)

目录 一、研究对象 二、理论推导 2.1 二维紧差分格式 2.2 紧交替方向格式 2.2.1 紧Peaceman-Rachford格式 2.2.2 紧D’Yakonov格式 2.2.3 紧Douglas格式 三、算例实现 四、结论 一、研究对象 继续以二维抛物型方程初边值问题为研究对象: 为了确保连续性,公式…

Vitis AI 环境搭建 KV260 PYNQ 安装 要点总结

目录 1. 环境 2. 工具及版本介绍 2.1 工具版本兼容性 2.2 DPU结构 2.3 DPU命名规则 3. Vitis AI 配置要点 3.1 配置安装 Docker 库 3.2 Install Docker Engine 3.3 添加 Docker 用户组并测试 3.4 克隆 Vitis AI 库 3.5 构建 Docker &#xff08;直接抓取&#xff09…

OpenHarmony 网络与连接—RPC连接

介绍 本示例使用ohos.rpc 相关接口&#xff0c;实现了一个前台选择商品和数目&#xff0c;后台计算总价的功能&#xff0c;使用rpc进行前台和后台的通信。 效果预览 使用说明&#xff1a; 点击商品种类的空白方框&#xff0c;弹出商品选择列表&#xff0c;选择点击对应的商品…

语音转换中的扩散模型——DDDM-VC

DDDM-VC: Decoupled Denoising Diffusion Models with Disentangled Representation and Prior Mixup for Verifed Robust Voice Conversion https://ojs.aaai.org/index.php/AAAI/article/view/29740https://ojs.aaai.org/index.php/AAAI/article/view/29740 1.概述 首先,语…

Python 中整洁的并行输出

原文&#xff1a;https://bernsteinbear.com/blog/python-parallel-output/ 代码&#xff1a;https://gist.github.com/tekknolagi/4bee494a6e4483e4d849559ba53d067b Python 并行输出 使用进程和锁并行输出多个任务的状态。 注&#xff1a;以下代码在linux下可用&#xff0c…

win10 系统怎么开启 guest 账户?

win10 系统怎么开启 guest 账户&#xff1f; 段子手168 前言&#xff1a; guest 账户即所谓的来宾账户&#xff0c;我们可以通过该账户访问计算机&#xff0c;如打印机共享等&#xff0c;但会在一定程度上受到限制。下面分享 WIN10 系统开启 guest 来宾账户的几种方法。 方法…

设备连接IoT云平台指南

一、简介 设备与IoT云间的通讯协议包含了MQTT&#xff0c;LwM2M/CoAP&#xff0c;HTTP/HTTP2&#xff0c;Modbus&#xff0c;OPC-UA&#xff0c;OPC-DA。而我们设备端与云端通讯主要用的协议是MQTT。那么设备端与IoT云间是如何创建通信的呢&#xff1f;以连接华为云IoT平台为例…

SpringBoot集成EasyExcel 3.x:高效实现Excel数据的优雅导入与导出

目录 介绍 快速开始 引入依赖 简单导出 定义实体类 自定义转换器 定义接口 测试接口 复杂导出 自定义注解 定义实体类 数据映射与平铺 自定义单元格合并策略 定义接口 测试接口 一对多导出 自定义单元格合并策略 测试数据 简单导入 定义接口 测试接口 参…