前言

  主流的Java Web服务器(Tomcat、Jetty、WebLogic、WebSphere等)都有多个自定义的类加载器以及具备的类加载优势,本篇博文主要是通过以Tomcat为例简单认识Java Web服务器具有的特点、以及自定义类加载器是如何实现的。此外,本篇博文主要是根据Tomcat 5.x类加载架构编写的。

  主要参考资料《深入理解Java虚拟机》


1、Web服务器需要解决的问题

  一个功能健全的Web服务器,都需要解决如下几个问题:

  (1)同一个服务器上的两个Web程序所使用的Java类库实现相互独立隔离:不能一个类库在同一个服务器上只有一份却被多个程序使用。

  (2)同一个服务器上的两个Web应用程序所使用的Java类库可以被共享:如果不能共享的话,虚拟机内部的Method Area 可能会出现过度膨胀现象。

  (3)服务器的类库要与Web应用程序的类库相互隔离,互不影响(即服务器不收Web程序的影响):很多服务器可能是用Java来实现,自然就得使用一些只属于自己的类库。

  (4)支持JSP的热替换(HotSwap):大部分主流服务器(WebLogic除外)修改JSP文件都不需要重启服务器。

  试想一下,如果上述的4点中任何一点不能满足,特别是(1)-(3)点,那么整个Java Web服务器都是“混乱”的,或者说是不健全的。

2、Tomcat类库存放目录

  既然要满足以上四点基本要求,那么当部署一个Web应用时,一个Classpath路径目录是无法满足需求的,通常会提供多个Classpath(一般以classes/lib命名)路径目录,而目录中都会有一个相应的自定义类加载器去加载里面的Java类。在Tomcat中就有三组目录,分别是:/common/*、/server/*、/shared/*;

  但值得强调的是在Tomcat 6.x之后这三组目录就不存在了,只有一个总的/lib目录,其他都需要进入配置文件需要时单独配置。所以这里介绍的目录未必能在tomcat安装目录下找到,不过其用意都是一样的;

  (1)/common/*:实际创建的时候为/common/lib/目录,可以被Tomcat与所有的Web应用共享的类库都在里面;

    但是在Tomcat 6之后/common/lib/目录被废弃,现在默认的是在/lib/下,实际上是将/common、/server、/shared三个目录默认合并到一个/lib目录下。如果不满足具体的业务需求的话,可以在conf/catalina.properties中设置server.loader与shared.loader。打开conf/catalina.properties可以看到:无论是${catalina.base},还是"${catalina.home},都会去找/lib/目录,指定加载的加载器。

    

    CATALINA_BASE:是实例配置位置,也就是一个tomcat可以配置多个实例,实例里面有自己的配置;

    CATALINA_HOME:是tomcat安装位置;

    common.loader、server.loader、shared.loade:表示tomcat指定自定义的三个加载器。

     

         

    

  (2)/server/*:实际创建的时候为/server/lib目录,只能被Tomcat使用的类库。

  (3)/shared/*:实际创建的时候为/shared/lib目录,可以被所有的Web应用程序共享使用,但是对Tomcat是不可见的。

  (4)/webapps(wtswebapps)/project/WEB-INF/*:实际为/WebApp/project/WEB-INF/lib,是Project独享有的类库。

3、Tomcat自定义加载器

  Tomcat自定义的类加载器主要有:ComnonClassLoader、CatalinaClassLoader、SharedClassLoader和WebAppClassLoader四个类加载器。

  (1)ComnonClassLoader:加载/common/*目录下的类,但是上述可知现在的指定加载都会在Catalina.properties中配置。

  (2)CatalinaClassLoader:加载/server/*目录下的类,同理现在在Catalina.properties中指定配置。

  (3)SharedClassLoader:加载/shared/*目录下的类,同理现在在Catalina.properties中指定配置。

  (4)WebAppClassLoader:加载/webapps(wtswebapps)/project/WEB-INF/*目录下项目lib中的类。

  结合之前三个类加载器(BootstrapClassLoader、ExtensionClassLoader、ApplicationLoader)与Tomcat自定义的四个加载器组合成的委派关系(Tomcat服务器的类加载机构)如下所示:

               

              (注:其中JasperLoader为Jsp类加载器,JSP文件编译后为Class文件,需要加载)

  通过此关系图我们可以得到如下两点:

  (1)CatalinaClassLoader与SharedClassLoader是相互隔离独立的;

  (2)WebAppClassLoader可以使用SharedClassLoader加载类,同理WebAppClassLoader可以代替JspClassLoader加载类,CommonClassLoader加载的类可以被CatalinaClassLoader与SharedClassLoader使用。

  同时需要注意还有以下两点:

  • JspClassLoader只能加载这个JSP文件所编译出Class文件,JSP文件被修改时,会替换当前JspClassLoader实例,之后会重新建立一个新的JspClassLoader实例,这也就是HotSwap的内部实现;
  • 不同的WebAppClassLoader是相互隔离的
  • 对于Tomcat 6.x版本,只有在conf/catalina.properties中配置了shared.loader与server.loader才能使用SharedClassLoader与CatalinaClassLoader实例,不然会默认使用ComnonClassLoader的实例来代替;

  • 如果默认设置不能满足需要,则可以通过修指定配置文件中的shared.loader与server.loader,之后重新启动Tomcat 5.x加载器架构。

  

 

 

Tomcat类加载架构的更多相关文章

  1. 服务器 : Apache Tomcat - 理解架构层次

    文章概览 相信很多接触java的人都对Tom猫有着多少的熟悉,就个人而言,本来只知道Tom简单的操作与配置,就像裹上一层纱,迷迷糊糊的. Tomcat的书籍本来就不多,高分的还是很久之前的版本,直到最 ...

  2. Tomcat整体架构分析

    下面让我们来看看Tomcat容器的整体结构: 本文的目的是覆盖这张图中所涉及的主要请求处理组件.而上图中的一些高级主题如集群和安全则不是在本文讨论的范围之内. 本图中,Service, Host, C ...

  3. 深入理解JVM虚拟机7:JNDI,OSGI,Tomcat类加载器实现

    打破双亲委派模型 JNDI JNDI 的理解   JNDI是 Java 命名与文件夹接口(Java Naming and Directory Interface),在J2EE规范中是重要的规范之中的一 ...

  4. java类加载器-Tomcat类加载器

    在上文中,已经介绍了系统类加载器以及类加载器的相关机制,还自定制类加载器的方式.接下来就以tomcat6为例看看tomat是如何使用自定制类加载器的.(本介绍是基于tomcat6.0.41,不同版本可 ...

  5. Tomcat 系统架构与设计模式

    Tomcat 系统架构与设计模式,第 1 部分: 工作原理 这个分为两个部分的系列文章将研究 Apache Tomcat 的系统架构以及其运用的很多经典设计模式.本文是第 1 部分,将主要从 Tomc ...

  6. 图解Tomcat类加载机制

    说到本篇的tomcat类加载机制,不得不说翻译学习tomcat的初衷. 之前实习的时候学习javaMelody的源码,但是它是一个Maven的项目,与我们自己的web项目整合后无法直接断点调试.后来同 ...

  7. Tomcat类加载器机制

    Tomcat为什么需要定制自己的ClassLoader: 1.定制特定的规则:隔离webapp,安全考虑,reload热插拔 2.缓存类 3.事先加载 要说Tomcat的Classloader机制,我 ...

  8. tomcat 系统架构与设计模式 第二部分 设计模式 转

    Tomcat 系统架构与设计模式,第 2 部分: 设计模式分析 许 令波, Java 开发工程师, 淘宝网 许令波,现就职于淘宝网,是一名 Java 开发工程师.对大型互联网架构设计颇感兴趣,并对一些 ...

  9. tomcat 系统架构与设计模式 第一部分 系统架构工作原理 转

    Tomcat 系统架构与设计模式,第 1 部分: 工作原理 许 令波, Java 开发工程师, 淘宝网 许令波,现就职于淘宝网,是一名 Java 开发工程师.对大型互联网架构设计颇感兴趣,并对一些开源 ...

随机推荐

  1. Python 协程 61

    什么是协程 协程,又称微线程,纤程.英文名Coroutine.一句话说明什么是线程:协程是一种用户态的轻量级线程. 协程的特点 协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到 ...

  2. elasticsearch中 refresh 和flush区别(转)

    elasticsearch中有两个比较重要的操作:refresh 和 flush refresh操作 当我们向ES发送请求的时候,我们发现es貌似可以在我们发请求的同时进行搜索.而这个实时建索引并可以 ...

  3. Maths | 层次分析法(Analytic Hierarchy Process)

    目录 1. 概述 2. AHP算法 2.1. 建立层级 2.2. 构造 成对 比较 矩阵 2.3. 成对比较矩阵的 一致性检验 与 层次单排序 2.4. 层次总排序 参考: (中文)https://z ...

  4. 3、MHC主要组织相容性复合体

    主要组织相容性复合体 (major histocompatibility complex MHC) 位于脊椎动物某对染色体上紧密连锁的基因群,其编码的蛋白是主要组织相容性抗原,是移植排斥反应的主要抗原 ...

  5. poj 1026

    这题一开始没看清楚 等级差距不超过1 1->2->3 就是错误的,因为3-1==2 ,意思是间接的也不行 其次等级最小是1,最大是n 你要到达1号首领的位置 假设1号等级x,限制m,最大上 ...

  6. SSAS 后端数据库访问模块中存在错误。 为绑定指定的大小太小,导致一个或多个列值被截断。

    在处理AS的过程中报错如上,经排查发现原因为数据库 “工号” 字段长度过长导致. 因为我的字段内容基本是人名加工号:张三/1001 不曾想有用户录入非正常数据 :张三/100/1001 这样导致我截取 ...

  7. Web前端JQuery面试题(二)

    Web前端JQuery面试题(二) 1.请写出jquery的语法? <script type="text/javascript"> $(document).ready( ...

  8. 线性整流函数(ReLU)

    线性整流函数(Rectified Linear Unit, ReLU),又称修正线性单元, 是一种人工神经网络中常用的激活函数(activation function),通常指代以斜坡函数及其变种为代 ...

  9. 一文了解Python中的循环(for while break continue 嵌套循环...)

    循环 目标 程序的三大流程 while 循环基本使用 break 和 continue while 循环嵌套 01. 程序的三大流程 在程序开发中,一共有三种流程方式: 顺序 —— 从上向下,顺序执行 ...

  10. Go语言下的线程模型

    阅读Go并发编程对go语言线程模型的笔记,解释的非常到,好记性不如烂笔头,忘记的时候回来翻一番,在此做下笔记. Go语言的线程实现模型,又3个必知的核心元素,他们支撑起了这个线程实现模型的主要框架: ...