双亲委派模型

本文深入解析Java中的双亲委派模型,解释其必要性并演示自定义类加载器的实现过程。了解如何避免安全风险,掌握findClass与defineClass等关键方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

其实,双亲委派模型并不复杂。自定义类加载器也不难!随便从网上搜一下就能搜出一大把结果,然后copy一下就能用。但是,如果每次想自定义类加载器就必须搜一遍别人的文章,然后复制,这样显然不行。可是自定义类加载器又不经常用,时间久了容易忘记。相信你经常会记不太清loadClassfindClassdefineClass这些函数我到底应该重写哪一个?它们主要是做什么的?本文大致分析了各个函数的流程,目的就是让你看完之后,难以忘记!或者说,延长你对自定义类加载器的记忆时间!随时随地想自定义就自定义!

1. 双亲委派模型

关于双亲委派模型,网上的资料有很多。我这里只简单的描述一下,就当是复习。

1.1 什么是双亲委派模型?

首先,先要知道什么是类加载器。简单说,类加载器就是根据指定全限定名称将class文件加载到JVM内存,转为Class对象。如果站在JVM的角度来看,只存在两种类加载器:

  • Bootstrap ClassLoader(启动类加载器)负责将%JAVA_HOME%/lib目录中或-Xbootclasspath中参数指定的路径中的,并且是虚拟机识别的(按名称)类库加载到JVM中
  • Extension ClassLoader(扩展类加载器)负责加载%JAVA_HOME%/lib/ext中的所有类库
  • Application ClassLoader(应用程序加载器) 负责ClassPath中的类库
  • 启动类加载器(Bootstrap ClassLoader):由C++语言实现(针对HotSpot),负责将存放在<JAVA_HOME>\lib目录或-Xbootclasspath参数指定的路径中的类库加载到内存中。

  • 其他类加载器:由Java语言实现,继承自抽象类ClassLoader。如:

    • 扩展类加载器(Extension ClassLoader):负责加载<JAVA_HOME>\lib\ext目录或java.ext.dirs系统变量指定的路径中的所有类库。
    • 应用程序类加载器(Application ClassLoader)。负责加载用户类路径(classpath)上的指定类库,我们可以直接使用这个类加载器。一般情况,如果我们没有自定义类加载器默认就是用这个加载器。

双亲委派模型工作过程是:如果一个类加载器收到类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器完成。每个类加载器都是如此,只有当父加载器在自己的搜索范围内找不到指定的类时(即ClassNotFoundException),子加载器才会尝试自己去加载。

类加载器的双亲委派模型

1.2 为什么需要双亲委派模型?

为什么需要双亲委派模型呢?假设没有双亲委派模型,试想一个场景:

黑客自定义一个java.lang.String类,该String类具有系统的String类一样的功能,只是在某个函数稍作修改。比如equals函数,这个函数经常使用,如果在这这个函数中,黑客加入一些“病毒代码”。并且通过自定义类加载器加入到JVM中。此时,如果没有双亲委派模型,那么JVM就可能误以为黑客自定义的java.lang.String类是系统的String类,导致“病毒代码”被执行。

而有了双亲委派模型,黑客自定义的java.lang.String类永远都不会被加载进内存。因为首先是最顶端的类加载器加载系统的java.lang.String类,最终自定义的类加载器无法加载java.lang.String类。

或许你会想,我在自定义的类加载器里面强制加载自定义的java.lang.String类,不去通过调用父加载器不就好了吗?确实,这样是可行。但是,在JVM中,判断一个对象是否是某个类型时,如果该对象的实际类型与待比较的类型的类加载器不同,那么会返回false。

举个简单例子:

ClassLoader1ClassLoader2都加载java.lang.String类,对应Class1、Class2对象。那么Class1对象不属于ClassLoad2对象加载的java.lang.String类型。

1.3 如何实现双亲委派模型?

双亲委派模型的原理很简单,实现也简单。每次通过先委托父类加载器加载,当父类加载器无法加载时,再自己加载。其实ClassLoader类默认的loadClass方法已经帮我们写好了,我们无需去写。

2. 自定义类加载器

2. 1几个重要函数

2.1.1 loadClass

loadClass默认实现如下:

public Class<?> loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
}

再看看loadClass(String name, boolean resolve)函数:

protected Class<?> loadClass(String name, boolean resolve)
    throws ClassNotFoundException
{
    synchronized (getClassLoadingLock(name)) {
        // First, check if the class has already been loaded
        Class c = findLoadedClass(name);
        if (c == null) {
            long t0 = System.nanoTime();
            try {
                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
            }

            if (c == null) {
                // If still not found, then invoke findClass in order
                // to find the class.
                long t1 = System.nanoTime();
                c = findClass(name);

                // this is the defining class loader; record the stats
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                sun.misc.PerfCounter.getFindClasses().increment();
            }
        }
        if (resolve) {
            resolveClass(c);
        }
        return c;
    }
}

从上面代码可以明显看出,loadClass(String, boolean)函数即实现了双亲委派模型!整个大致过程如下:

  1. 首先,检查一下指定名称的类是否已经加载过,如果加载过了,就不需要再加载,直接返回。
  2. 如果此类没有加载过,那么,再判断一下是否有父加载器;如果有父加载器,则由父加载器加载(即调用parent.loadClass(name, false);).或者是调用bootstrap类加载器来加载。
  3. 如果父加载器及bootstrap类加载器都没有找到指定的类,那么调用当前类加载器的findClass方法来完成类加载。

话句话说,如果自定义类加载器,就必须重写findClass方法!

2.1.1 find Class

findClass的默认实现如下:

protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
}

可以看出,抽象类ClassLoaderfindClass函数默认是抛出异常的。而前面我们知道,loadClass在父加载器无法加载类的时候,就会调用我们自定义的类加载器中的findeClass函数,因此我们必须要在loadClass这个函数里面实现将一个指定类名称转换为Class对象.

如果是是读取一个指定的名称的类为字节数组的话,这很好办。但是如何将字节数组转为Class对象呢?很简单,Java提供了defineClass方法,通过这个方法,就可以把一个字节数组转为Class对象啦~

2.1.1 defineClass

defineClass主要的功能是:

将一个字节数组转为Class对象,这个字节数组是class文件读取后最终的字节数组。如,假设class文件是加密过的,则需要解密后作为形参传入defineClass函数。

defineClass默认实现如下:

protected final Class<?> defineClass(String name, byte[] b, int off, int len)
        throws ClassFormatError  {
        return defineClass(name, b, off, len, null);
}

2.2 函数调用过程

上一节所提的函数调用过程如下:

自定义函数调用过程

2.3 简单示例

首先,我们定义一个待加载的普通Java类:Test.java。放在com.huachao.cl包下:

package com.huachao.cl;

public class Test {
    public void hello() {
        System.out.println("恩,是的,我是由 " + getClass().getClassLoader().getClass()
                + " 加载进来的");
    }
}

注意:

如果你是直接在当前项目里面创建,待Test.java编译后,请把Test.class文件拷贝走,再将Test.java删除。因为如果Test.class存放在当前项目中,根据双亲委派模型可知,会通过sun.misc.Launcher$AppClassLoader 类加载器加载。为了让我们自定义的类加载器加载,我们把Test.class文件放入到其他目录。

在本例中,我们Test.class文件存放的目录如下:

class文件目录

接下来就是自定义我们的类加载器:

import java.io.FileInputStream;
import java.lang.reflect.Method;

public class Main {
    static class MyClassLoader extends ClassLoader {
        private String classPath;

        public MyClassLoader(String classPath) {
            this.classPath = classPath;
        }

        private byte[] loadByte(String name) throws Exception {
            name = name.replaceAll("\\.", "/");
            FileInputStream fis = new FileInputStream(classPath + "/" + name
                    + ".class");
            int len = fis.available();
            byte[] data = new byte[len];
            fis.read(data);
            fis.close();
            return data;

        }

        protected Class<?> findClass(String name) throws ClassNotFoundException {
            try {
                byte[] data = loadByte(name);
                return defineClass(name, data, 0, data.length);
            } catch (Exception e) {
                e.printStackTrace();
                throw new ClassNotFoundException();
            }
        }

    };

    public static void main(String args[]) throws Exception {
        MyClassLoader classLoader = new MyClassLoader("D:/test");
        Class clazz = classLoader.loadClass("com.huachao.cl.Test");
        Object obj = clazz.newInstance();
        Method helloMethod = clazz.getDeclaredMethod("hello", null);
        helloMethod.invoke(obj, null);
    }
}

最后运行结果如下:

恩,是的,我是由 class Main$MyClassLoader 加载进来的
<think>我们正在讨论Java中的双亲委派模型及其工作原理。根据引用内容,双亲委派模型是Java类加载机制中的一种重要模型,它通过特定的层次结构和委派机制来确保类的唯一性和安全性。首先,类加载器之间的层次关系被称为双亲委派模型。该模型要求除了顶层的启动类加载器外,其余的类加载器都应该有自己的父类加载器(通过组合关系实现)。类加载器在加载类时,首先会委托给父类加载器去尝试加载,只有当父类加载器无法完成加载时,才由自己来加载。双亲委派模型的工作过程如下:1.当一个类加载器收到类加载请求时,它不会自己先尝试加载,而是将这个请求委派给父类加载器完成。2.每一层的类加载器都是如此,因此所有的加载请求最终都会传送到顶层的启动类加载器。3.只有当父类加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。双亲委派模型好处是:-确保核心库的类型安全:例如`java.lang.Object`类,无论哪一个类加载器要加载这个类,最终都会委派给启动类加载器加载,这样就保证了Object类在各种类加载器环境中都是同一个类。-避免重复加载:父类加载器已经加载的类,子类加载器不会再加载。-防止核心API被篡改:如果没有双亲委派模型,用户自定义了一个`java.lang.Object`类,那么系统中就会出现多个不同的Object类,导致混乱。在Java中,类加载器分为几种(引用中提到了四种,但通常我们分为三种):1.启动类加载器(BootstrapClassLoader):加载JAVA_HOME/lib目录下的核心类库,如rt.jar。它是虚拟机的一部分,由C++实现,不是Java类。2.扩展类加载器(ExtensionClassLoader):负责加载JAVA_HOME/lib/ext目录下的扩展类。3.应用程序类加载器(ApplicationClassLoader):也称为系统类加载器,负责加载用户类路径(ClassPath)上的类库。另外,用户也可以自定义类加载器。双亲委派模型的实现代码(在`java.lang.ClassLoader`中)主要体现在`loadClass`方法中。下面是该方法的一个简化描述:```javaprotectedClass<?>loadClass(Stringname,booleanresolve)throwsClassNotFoundException{synchronized(getClassLoadingLock(name)){//首先,检查类是否已经被加载Class<?>c=findLoadedClass(name);if(c==null){try{if(parent!=null){//如果有父加载器,则委派给父加载器加载c=parent.loadClass(name,false);}else{//如果没有父加载器,则委派给启动类加载器c=findBootstrapClassOrNull(name);}}catch(ClassNotFoundExceptione){//父加载器无法加载,捕获异常但不处理,继续往下执行}if(c==null){//如果父加载器没有加载到,则调用自己的findClass方法进行加载c=findClass(name);}}if(resolve){resolveClass(c);}returnc;}}```在双亲委派模型中,类加载器在加载类时,会先递归地委托给父类加载器,直到启动类加载器。如果父类加载器不能加载,子类加载器才会尝试加载。这样保证了同一个类不会被不同的类加载器重复加载,从而保证了类的唯一性。此外,每个类加载器都有一个独立的类名称空间。因此,两个类是否相等,不仅取决于类本身,还取决于加载它们的类加载器。也就是说,同一个类文件被不同的类加载器加载,它们就是不同的类。双亲委派模型是Java安全模型的一部分,但也存在一些缺陷。例如,基础类型需要调用回用户的代码(如SPI服务发现机制),这时就需要打破双亲委派模型。Java提供了线程上下文类加载器(ThreadContextClassLoader)来解决这类问题。总结:双亲委派模型是Java类加载机制的核心,它通过层次化的委派机制保证了类的唯一性和安全性,避免核心类被篡改,并防止了重复加载。</think>### Java双亲委派模型工作原理详解 #### **一、双亲委派模型的核心机制** 双亲委派模型是Java类加载的核心规则,其工作流程如下: 1. **层次结构** - 类加载器分为四层(由高到低): - **启动类加载器**(Bootstrap ClassLoader):加载`JAVA_HOME/lib`核心库(如`rt.jar`) - **扩展类加载器**(Extension ClassLoader):加载`JAVA_HOME/lib/ext`扩展库 - **应用程序类加载器**(Application ClassLoader):加载用户类路径(ClassPath)的类 - **自定义类加载器**:用户实现的加载器 - 父子关系通过**组合**实现(非继承),子加载器持有父加载器的引用[^2] 2. **委派流程** ```mermaid graph TD A[子类加载器收到请求] --> B{是否已加载?} B -- 是 --> C[返回已加载类] B -- 否 --> D[委派父加载器] D --> E{父加载器能否完成?} E -- 能 --> F[返回父加载结果] E -- 否 --> G[子加载器自行加载] ``` - **关键规则**:子加载器必须先将请求委派给父加载器,只有父加载器无法完成时(如不在其搜索范围),子加载器才会尝试加载[^1][^2] #### **二、工作原理解析** 以加载`java.lang.Object`为例: 1. 用户自定义加载器收到请求 → 委派给应用程序类加载器 2. 应用程序类加载器委派给扩展类加载器 3. 扩展类加载器委派给启动类加载器 4. 启动类加载器在`rt.jar`中找到并加载`Object`类 → 逐层返回结果 $$ \text{加载路径:} \text{自定义} \to \text{应用} \to \text{扩展} \to \text{启动} $$ #### **三、核心价值与优势** 1. **安全性保障** - 防止核心API被篡改(如用户伪造`java.lang.Object`类)[^1] - 确保核心类库由可信加载器加载 2. **类唯一性保证** - 类的身份由**加载器+类全名**共同确定 - 不同加载器加载的相同类文件,JVM视为不同类[^3] ```java // 示例:不同加载器加载的String比较 Class<?> strClass1 = loader1.loadClass("java.lang.String"); Class<?> strClass2 = loader2.loadClass("java.lang.String"); System.out.println(strClass1 == strClass2); // 输出false ``` 3. **避免重复加载** - 父加载器已加载的类,子加载器不会重复加载 #### **四、关键代码实现** 在`ClassLoader.loadClass()`中的逻辑: ```java protected Class<?> loadClass(String name, boolean resolve) { synchronized (getClassLoadingLock(name)) { // 1. 检查是否已加载 Class<?> c = findLoadedClass(name); if (c == null) { try { // 2. 委派父加载器 if (parent != null) { c = parent.loadClass(name, false); } else { c = findBootstrapClassOrNull(name); } } catch (ClassNotFoundException e) { // 父加载器无法完成 } // 3. 自行加载 if (c == null) { c = findClass(name); } } return c; } } ``` #### **五、打破双亲委派的场景** 在特定场景需绕过该模型: 1. **SPI机制**(如JDBC驱动加载) - 使用线程上下文类加载器(Thread Context ClassLoader) 2. **热部署需求** - 自定义加载器重写`loadClass()`方法 3. **OSGi模块化** - 实现网状加载结构 > **本质矛盾**:基础类需调用用户代码时,必须打破委派链[^4] --- ### 相关问题 1. **双亲委派模型如何解决类冲突问题?** *例如用户自定义了`java.lang.String`类会怎样?* 2. **哪些场景需要打破双亲委派模型?** *请结合JDBC驱动加载的实例说明* 3. **自定义类加载器的实现要点是什么?** *需要重写哪些关键方法?* 4. **不同类加载器加载的类如何实现隔离?** *解释JVM中"类全名+加载器=唯一标识"的原理* 5. **双亲委派模型在模块化系统(如JPMS)中有何变化?** *比较与传统模型的差异* [^1]: 双亲委派模型确保核心类由启动加载器加载,避免用户篡改核心类库 [^2]: 类加载器通过组合实现父子关系,形成层级委派链 [^3]: 类在JVM的唯一性由类加载器和类全名共同决定 [^4]: `ClassLoader`的`loadClass()`和`findClass()`是扩展关键点
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值