Java的类加载机制之双亲委派模型为什么叫双亲委派?

前言

学习java类加载机制是,常常因为各种奇怪名词而苦恼,比如这个双亲委派就让人非常疑惑,为什么叫双亲委派?
Java类加载器的工作模式中,类加载请求沿着类加载器的层级结构,从下至上(即从子类加载器到父类加载器)逐级传递。这里的“双亲”实际上指的是类加载器的父加载器,因为在类加载器的层级结构中,除了顶层的启动类加载器外,每个类加载器都有一个父加载器(那也是单亲啊(bushi))。

打个比方,“委派”就像是孩子遇到问题时,不是自己立即尝试解决,而是先问父母(即父类加载器),如果父母能解决就由他们处理;如果父母也不知道怎么办,再由孩子自己尝试解决。因此,这种类加载的层级传递和责任委托的方式,就被形象地称为“双亲委派模型”。

双亲委派模型(Parent Delegation Model)是Java类加载机制中非常重要的一部分,用于确保类的安全加载。这一机制主要体现在Java虚拟机(JVM)的类加载器层级结构中。在Java中,类加载器并不是孤立存在的,而是存在层次关系,大致可以分为启动类加载器(Bootstrap ClassLoader)、扩展类加载器(Extension ClassLoader)和应用程序类加载器(Application ClassLoader,也称系统类加载器),以及可能由应用程序自定义的类加载器。

双亲委派模型的工作流程:

  1. 当一个类加载器收到类加载请求时,它首先不会自己尝试去加载这个类,而是将这个请求委托给它的父类加载器去完成。

  2. 如果父类加载器还存在其自身的父类加载器,则进一步向上委托,直至委托到启动类加载器。因为启动类加载器是所有类加载器的最顶端,它负责加载Java的核心库(如rt.jar中的类)。

  3. 如果父类加载器无法处理这个加载请求(即在其搜索范围内未找到所需的类),那么子类加载器会尝试自己加载这个类。

双亲委派流程图:

在Java中,ClassLoader类的loadClass(String name)方法实现了双亲委派的具体逻辑。当一个类加载器接收到加载请求时,它会先调用父加载器的loadClass()方法,如果父加载器加载成功,则直接返回;否则,再尝试自己加载。
在这里插入图片描述

目的和优势:

  • 防止类的重复加载:确保每个类在整个应用程序中只被加载一次,有助于节省内存,并且避免了因重复加载而导致的类定义不一致问题。
  • 保证核心库安全:通过委托给启动类加载器加载Java核心API,可以防止用户自定义类假冒核心类库的类,增强了系统的安全性和稳定性。
  • 支持模块化和插件化:不同的类加载器可以为不同的模块或插件服务,它们之间互不影响,这有利于实现程序的模块化和热部署。

自定义双亲委派模型:

双亲委派的核心逻辑实现在java.lang.ClassLoaderloadClass(String name)方法中,但通常建议重写findClass(String name)方法来完成自定义加载逻辑,以保持双亲委派的机制。

public class MyClassLoader extends ClassLoader {
    
    private String classPath; // 自定义类加载路径
    
    public MyClassLoader(String classPath) {
        this.classPath = classPath;
    }
    
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] classData = getClassData(name);
        if (classData == null) {
            throw new ClassNotFoundException();
        } else {
            return defineClass(name, classData, 0, classData.length);
        }
    }
    
    private byte[] getClassData(String className) {
        // 实现从classPath中读取类的二进制数据逻辑
        // 这里省略具体实现细节
    }

    // 重写loadClass方法以维持双亲委派模型
    @Override
    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
        synchronized (getClassLoadingLock(name)) {
            // 首先检查是否已被加载
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t1 = System.nanoTime();
                try {
                    // 如果父类加载器不为空,尝试通过父类加载
                    if (getParent() != null) {
                        c = getParent().loadClass(name);
                    } else {
                        // 如果父加载器为空,使用启动类加载器加载
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // 如果父加载器找不到,才由当前类加载器尝试加载
                    c = findClass(name);
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }
}

以上代码,当调用MyClassLoaderloadClass方法加载一个类时,它首先会委托给父类加载器尝试加载,如果父类加载器加载不到(抛出ClassNotFoundException),才会尝试自己加载。这就是双亲委派模型的核心逻辑。

  • 19
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java类加载器采用双亲委派机制,这是Java安全模型的重要组成部分。这种机制保证Java核心API不会被随意篡改,同时也保证了Java程序的稳定性和安全性。 双亲委派机制的基本原则是:当一个类加载器接收到类加载请求时,它首先将该请求委派给它的父类加载器去完成,直到最顶层的父类加载器。只有在父类加载器无法完成类加载任务时,才由子类加载器自行加载。这种机制确保了Java核心API的安全性,因为只有Bootstrap ClassLoader能够加载Java核心API,其他类加载器都无法篡改这些类。 在双亲委派机制,每个类加载器都有一个父类加载器。如果一个类加载器需要加载某个类,它会先委托给它的父类加载器去加载。如果父类加载器无法加载该类,才会由该类加载器自己去加载。这样一来,如果一个类已经被加载了,那么其类加载器的父类加载器肯定已经加载了该类,因此不会重复加载,也就避免了类的重复加载。 双亲委派机制的实现是通过ClassLoader类的loadClass()方法实现的。这个方法首先检查是否已经加载了该类,如果已经加载了就直接返回,否则就委托给父类加载器去加载。如果父类加载器无法加载该类,就调用findClass()方法自己加载。这样一来,每个类加载器都只需要实现自己的findClass()方法,而loadClass()方法则由ClassLoader类统一实现,从而实现了双亲委派机制

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值