开头说两句

小刀博客: http://www.lixiang.red
小刀公众号: 程序员学习大本营

学习背景

上期我们聊到了tomcat中各个组件在默认值,在其中,我们看到了有关类加载器的代码, 如context中初始化wabAppLoader
https://www.lixiang.red/articles/2019/08/09/1565361802994.html
今天我们一起学习类加载器相关知识

java里面的类加载器

我们在写java代码时,源文件是 *.java , 然后经过编译之后,会变成 .class 文件,类加载器加载的,实际上就是.class文件, 在实际开发中,我们会把相关的 .class 文件,打成一个jar包, 然后直接加载jar包就可以了.
类加载器就是用来加载这些类到虚拟机里,供程序调用

Bootstrap Class Loader

用来加载JVM提供的基础运行类,即位于%JAVA_HOME%jre/lib 这个目录下面的核心类库

Extension Class Loader

java提供的一个标准的扩展机制,用于加载除核心类库外的jar包.默认的扩展目录是 %JAVA_HOME%/jar/lib/ext

该目录下的类库,对所有基于该JVM运行的程序都是可见的

System class loader

用于加载环境变量 CLASSPATH 指定目录下的或者是用 -classpath运行参数指定的jar包.

System Class Loader 通常用于加载应用程序jar包及其启动入口类(Tomcat Bootstrap类就是由System Class Loader 来加载的)

类加载器的双亲委派模式

上面三种类加载器,实际上是有父子关系,Bootstrap 是 Extension的父加载器, Extension 是System的父加载器
当System ClassLoader 拿到一个class 文件之后, 会先问父加载器(Extension Class Loader)能不能加载,当(Extension )接收到请求时,会先问问他的父加载器(BootStrap类加载器能不能加载). 如果 Bootstrap可以加载,则由Bootstrap来加载,如不能,则由Extension来加载. 如果 Extension 也加载不了的话,最后由System 类加载器来加载

tomcat中的类加载器

总共有四种类加载器: Common Class Loader, Catalina Class Loader , Shared Class Loader, Web AppClass Loader.
tomcat中各个类加载器初始化,声明的地方

/**     * 初始化tomcat中的三大类加载器     */    private void initClassLoaders() {        try {            // CommonLoader 默认继承AppClassLoader            commonLoader = createClassLoader("common", null);            if( commonLoader == null ) {                // no config file, default to this loader - we might be in a 'single' env.                commonLoader=this.getClass().getClassLoader();            }            // 以CommonLoader 为父加载器            catalinaLoader = createClassLoader("server", commonLoader);            // 以CommonLoader 为父加载器            sharedLoader = createClassLoader("shared", commonLoader);        } catch (Throwable t) {            handleThrowable(t);            log.error("Class loader creation threw exception", t);            System.exit(1);        }    }

Common Class loader

以System Class Loader 为父类加载器, 是位于Tomcat 应用服务器顶层的公用类加载器,默认是加载$CATALINE_HOME/lib 下的jar

Catalina Class Loader

以Common Class Loader 为父加载器.用于加载 Tomcat 应用服务器本身的.可以在下图中看到使用的位置
1.设置当前线程的类加载器为Catalina Class Loader , 在没设置之前,是由 shell 脚本用 System Class Loader 来加载的
2. 用Catalina Class Loader 加载 Catalina.class 这个文件,并完成一系统组件的初始

shared Class Loader

以Common 为父加载器,是所有web应用的父加载器
使用位置如下

/**     * Set the shared extensions class loader.     *     * @param parentClassLoader The shared extensions class loader.     */    public void setParentClassLoader(ClassLoader parentClassLoader) {        this.parentClassLoader = parentClassLoader;    }

在源码中声明parentClassLoader时有一个小坑,他默认是声明的加载Catalina的类载器            即:Catalina Class Loader,但实际上,在实例化后,我们会用反射调用其setParentClassLoader 方法,将parentClassLoader 更改为shared Class Loader

// XXX Should be moved to embedded    /**     * The shared extensions class loader for this server.     */    protected ClassLoader parentClassLoader =        Catalina.class.getClassLoader();

使用地点、

WebApp Class Loader

初始化的地点有两处:
1.createStartDigester中

digester.addObjectCreate(prefix + "Context/Loader",                            "org.apache.catalina.loader.WebappLoader",                            "className");
  1. StandardContext.startInternal 方法中

    if (getLoader() == null) {            WebappLoader webappLoader = new WebappLoader(getParentClassLoader());            webappLoader.setDelegate(getDelegate());            setLoader(webappLoader);        }

其作用是,每个独立的Context(web应用)都使用独立的ClassLoader,加载我们web应用中,WEB-INFO/libs 这个目录下的jar(如我们在应用中引用的spring , mybatis 这些包)
这个做的好处是,不同的web应用包不会冲突,如A应用用的是spring 4.X , B应用用的是spring 5.X , 他们可以在同一个tomcat中运行

最后说两句

tomcat的类加载机制, 是开始的一个比较复杂的点,需要好好理一理,边看代码边做笔记,这个类加载器什么时候初始化的,做了什么,然后在哪里使用的. 大家在学习过种中,有什么问题,可以和小刀一起交流讨论: best396975802

tomcat jar包_tomcat学习|tomcat中的类加载器相关推荐

  1. tomcat jar包编译后变成文件夹_tomcat学习|tomcat中的类加载器

    开头说两句 小刀博客: https://www.lixiang.red 小刀公众号: 程序员学习大本营 学习背景 上期我们聊到了tomcat中各个组件在默认值,在其中,我们看到了有关类加载器的代码, ...

  2. java spring包_java 自定义加载器,加载spring包,动态加载实现,jar包隔离,tomcat加载webapp方式...

    java 自定义加载器,加载spring包,动态加载实现,jar包隔离,tomcat加载webapp方式 发布时间:2018-08-20 12:02, 浏览次数:774 , 标签: java spri ...

  3. k8s部署jar包_学习K8S之路.6--- 在K8S中部署Jenkins,并使用Jenkins打包jar包

    一:部署jenkins jenkins官网:https://jenkins.io/download/ jenkins镜像:https://hub.docker.com/r/jenkins/jenkin ...

  4. TOPGP5.3:导入jar包并在4GL中引用

    查看环境$CLASSPATH 上传引用到的JAR包到以下目录 /u1/topprod/tiptop/ds4gl2/bin/javaad/jar 以上为GP5.3目录,其他版本系统可根据查看到的$CLA ...

  5. jmeter导入jar包后在beanshell中import失效的问题解决

    jmeter导入jar包后在beanshell中import失效的问题解决 参考文章: (1)jmeter导入jar包后在beanshell中import失效的问题解决 (2)https://www. ...

  6. 随笔18 java中的类加载器

    类的加载是由类加载器完成的,类加载器包括:根加载器( BootStrap ).扩展加载器( Extension ).系统加载器( System )和用户自定义类加载器( java.lang.Class ...

  7. 通过这一篇文章,可以把Java中的类加载器了解的七七八八了

    对于每个开发人员来说,java.lang.ClassNotFoundExcetpion这个异常几乎都遇到过,而追求其该异常的来源的话,就免不了谈一谈Java的类加载器了.本文就基于启动类加载器.扩展类 ...

  8. 【Android 逆向】加壳的 Android 应用启动流程 | 使用反射替换 LoadedApk 中的类加载器流程

    文章目录 一.加壳的 Android 应用启动流程 二.使用反射替换 LoadedApk 中的类加载器流程 一.加壳的 Android 应用启动流程 加壳的 Android 应用启动流程 : 加壳的 ...

  9. jdk类加载器执行步骤_[JDK 11] jcmd中的类加载器层次结构详细信息

    jdk类加载器执行步骤 自从在JavaOne 2012上听说 jcmd以来,我就一直是命令行诊断工具jcmd的 粉丝 . 从那时起,我已经广泛使用了此工具,并多次在此博客中发布过此工具: jcmd:一 ...

最新文章

  1. c语言编程用得上i7,为什么我的C应用程序比Core i7上的C应用程序(使用相同的库)更快...
  2. sqlplus连接远程数据库
  3. kotlin学习笔记——接口与委托
  4. UI设计素材|APP引导页简约风格特征
  5. css top 定位百分比,css 使用relative设置top为百分比值的方法(仿百度首页)
  6. 通用权限管理平台--数据模型定义
  7. 给各位读者朋友们、热心访问本博客的朋友们的一份新年贺卡
  8. 刚刚编译运行的项目,突然就不能用了。AndroidStudio最难用
  9. 玩转安卓10源码开发定制(17)编译Windows平台adb和fastboot工具
  10. Java系统日志管理
  11. 本科计算机的逻辑学,逻辑学本科专业介绍
  12. 花几千块钱,线上培训软件测试有用吗?
  13. light动名词_英语语法大全之动名词
  14. Android 11 状态栏电池图标的定制
  15. slam十四讲,ch5joinmap报错,已解决
  16. java中的静态是什么?怎么用?
  17. 如何统一管理谷歌GKE、AWS EKS和Oracle OKE
  18. Linux 串口工具 cutecom
  19. Matlab中angle函数使用
  20. yyyyyyyyyyyyyy

热门文章

  1. ES6中解构赋值深入解读
  2. python如何计算字符宽度_Python中计算字符宽度
  3. 切换node版本 nvm 的基本使用 -- 以及安装公司特有的镜像源
  4. .gitignore失效 无法忽略node_modules问题
  5. jade模板引擎修改为ejs模板引擎
  6. 遍历一个数据去掉最后一个元素的样式
  7. ES6公用分页组件的封装及应用举例
  8. 用java实现验证码(CAPTCHA)
  9. longest-repeating-character-replacement(难)
  10. 8.5 趣味游戏(2)