Tomcat ClassLoader打破双亲委派源码解析

 更新时间:2023年09月01日 11:31:34   作者:汤卜  
这篇文章主要为大家介绍了Tomcat ClassLoader打破双亲委派源码解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

java加载类

java加载类的时候需要使用类加载器,开发人员可以定制类的加载器,比如tomcat就扩展了自己的类加载器。这篇文章结合代码研究一下jdk类的加载器和tomcat的类加载

先上图复习一下

来看一下jdk的ClassLoader的代码

protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // 本类加载器从缓存里面拿取已经加载过的ClassLoader
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    // 父类为BootStrapClassLoader的classLoader 的parent变量为空
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
                ..........
    }

注意看一下核心代码,实际上parent只要不为空就由parent来加载,因此当类未加载的时候,总是由上级来优先加载。

tomcat的类加载

首先来看一下tomcat的类加载器,tomcat写了一个自己的类加载器,看下基类WebappClassLoaderBase的实现

@Override
    public Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
           ................
            // 从缓存里面取,不影响加载顺序,请着重忽略
            clazz = findLoadedClass0(name);
            ..........
            clazz = findLoadedClass(name);
            .........
           /**
     * The bootstrap class loader used to load the JavaSE classes. In some
     * implementations this class loader is always <code>null</code> and in
     * those cases {@link ClassLoader#getParent()} will be called recursively on
     * the system class loader and the last non-null result used.
     */
            // 注意看注释,javaseLoader是bootstrap加载的,所以javaseLoader是扩展类加载器
            ClassLoader javaseLoader = getJavaseClassLoader();
            clazz = javaseLoader.loadClass(name);
            boolean delegateLoad = delegate || filter(name, true);
            // (2) Search local repositories
            if (log.isDebugEnabled()) {
                log.debug("  Searching local repositories");
            }
            try {
            // 重写了findClass方法,从项目中的lib和class里加载
                clazz = findClass(name);
                if (clazz != null) {
                    if (log.isDebugEnabled()) {
                        log.debug("  Loading class from local repository");
                    }
                    if (resolve) {
                        resolveClass(clazz);
                    }
                    return clazz;
                }
            } catch (ClassNotFoundException e) {
                // Ignore
            }
            // (3) Delegate to parent unconditionally
            if (!delegateLoad) {
                if (log.isDebugEnabled()) {
                    log.debug("  Delegating to parent classloader at end: " + parent);
                }
                try {
//  继承于SystemClassLoader该类,所以这个parent是SystemClassLoader,
                    clazz = Class.forName(name, false, parent);
                    if (clazz != null) {
                        if (log.isDebugEnabled()) {
                            log.debug("  Loading class from parent");
                        }
                        if (resolve) {
                            resolveClass(clazz);
                        }
                        return clazz;
                    }
                } catch (ClassNotFoundException e) {
                    // Ignore
                }
            }
        }
        throw new ClassNotFoundException(name);
    }

把顺序相关外的代码省略了,注意看注释。

因此来梳理一下加载顺序

1.找ext加载器,ext会优先走BootStrap加载器,加载java核心类库

2.ext加载不到就去走重写的findClass,走web项目的lib包和class目录,加载项目的class

3.重写的findClass找不到,就去系统类加载器SystemClassLoader来加载。

tomcat为什么要重写这个ClassLoader呢

1.最重要的就是做隔离性,不同的 web包引入不同的jar包版本,如果重写类的加载顺序,统一走SystemClassloader加载,那么只会加载一个类在内存,另外一个版本的class永远也不会加载进内存。无法实现web项目的隔离

2.可以针对一个应用进行重新部署,一个应用一个加载器,需要重新部署,就把这个classLoader加载的类重新加载就可以了

所以看起来打破双亲委派这种大名头的知识,看一下源码就清晰了,养成阅读源码习惯就可以

以上就是Tomcat ClassLoader打破双亲委派源码解析的详细内容,更多关于Tomcat ClassLoader双亲委派的资料请关注脚本之家其它相关文章!

相关文章

  • Java 添加、删除、替换、格式化Word中的文本的步骤详解(基于Spire.Cloud.SDK for Java)

    Java 添加、删除、替换、格式化Word中的文本的步骤详解(基于Spire.Cloud.SDK for Java)

    这篇文章主要介绍了Java 添加、删除、替换、格式化Word中的文本(基于Spire.Cloud.SDK for Java),本文分步骤通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08
  • Java中JSON字符串与java对象的互换实例详解

    Java中JSON字符串与java对象的互换实例详解

    这篇文章主要介绍了在java中,JSON字符串与java对象的相互转换实例详解,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-08-08
  • Spring Cloud Consul的服务注册与发现

    Spring Cloud Consul的服务注册与发现

    这篇文章主要介绍了Spring Cloud Consul服务注册与发现的实现方法,帮助大家更好的理解和学习使用spring框架,感兴趣的朋友可以了解下
    2021-02-02
  • Java中String使用方法及分析(UTF-8简单编码/解码器实现)

    Java中String使用方法及分析(UTF-8简单编码/解码器实现)

    在Java中String类是用来表示和操作字符串的,它是Java中最常用的类之一,提供了丰富的方法来创建、操作和查询字符串,这篇文章主要介绍了Java中String使用方法及分析(UTF-8简单编码/解码器实现)的相关资料,需要的朋友可以参考下
    2025-12-12
  • 详解Java从工厂方法模式到 IOC/DI思想

    详解Java从工厂方法模式到 IOC/DI思想

    工厂方法(Factory Method)模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中。核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口。本文将详细介绍Java从工厂方法模式到 IOC/DI思想。
    2021-06-06
  • 基于UDP协议实现聊天系统

    基于UDP协议实现聊天系统

    这篇文章主要为大家详细介绍了基于UDP协议实现聊天系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-04-04
  • JAVA读取文本文件内容实例代码

    JAVA读取文本文件内容实例代码

    这篇文章主要给大家介绍了关于JAVA读取文本文件内容的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Java具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-09-09
  • 关于SpringBoot拦截器拦截静态资源的问题

    关于SpringBoot拦截器拦截静态资源的问题

    这篇文章主要介绍了关于SpringBoot拦截器拦截静态资源的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • SpringBoot 整合Mybatis-Plus并输出SQL日志示例详解

    SpringBoot 整合Mybatis-Plus并输出SQL日志示例详解

    这篇文章主要介绍了SpringBoot整合Mybatis-Plus并输出SQL日志,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-06-06
  • Spring中XmlWebApplicationContext的实现

    Spring中XmlWebApplicationContext的实现

    XmlWebApplicationContext是Spring Framework中的一个重要类,本文主要介绍了Spring中XmlWebApplicationContext,具有一定的参考价值,感兴趣的可以了解一下
    2024-08-08

最新评论