首页 > 其他分享 >【JVM】详解双亲委派机制

【JVM】详解双亲委派机制

时间:2024-06-18 12:28:02浏览次数:20  
标签:委派 name 详解 双亲 JVM classData 机制 加载

双亲委派机制是Java类加载器的一种工作模式,确保类加载的一致性和安全性。以下是详细的定义、优缺点以及如何破坏双亲委派机制的描述。

双亲委派机制的定义

双亲委派机制(Parent Delegation Model)是一种类加载器的工作模式。在这种模式下,类加载器在加载类时,会先将加载请求委派给父类加载器处理,只有当父类加载器无法完成类加载时,子类加载器才会尝试加载该类。

双亲委派机制的工作过程:
  1. 检查缓存:类加载器首先检查是否已经加载过该类,如果已经加载过,则直接返回该类。
  2. 委派父加载器:如果没有加载过该类,类加载器将加载请求委派给父类加载器。
  3. 递归过程:父类加载器也按照相同的过程处理加载请求,直到委派到启动类加载器(Bootstrap ClassLoader)。
  4. 启动类加载器加载:启动类加载器尝试加载类,如果加载成功,则返回该类,否则继续步骤5。
  5. 逐级回退:如果启动类加载器无法加载该类,加载请求逐级回退到下一级类加载器,直至回退到最初的请求者。
  6. 自行加载:最终,如果所有父类加载器都无法加载该类,子类加载器才会尝试自行加载。

双亲委派机制的优点

  1. 安全性:双亲委派机制确保了核心类库(如java.lang.*包)不会被重复加载和篡改。只有启动类加载器能加载这些核心类库,避免了安全风险。
  2. 避免类重复加载:通过委派机制,保证了类只会被加载一次,从而节省内存,提高性能。
  3. 一致性:同一个类在整个Java应用中只有一个唯一的定义,避免了同名类不同实现的冲突问题。

双亲委派机制的缺点

  1. 灵活性不足:双亲委派机制的严格层级关系使得子类加载器很难绕过父类加载器直接加载类,这在某些情况下限制了灵活性。
  2. 复杂性:实现自定义类加载器时,需要理解和实现双亲委派机制,这增加了开发的复杂性。

如何破坏双亲委派机制

破坏双亲委派机制通常是为了满足某些特定需求,如加载自定义的类或版本不同的类库。以下是一些破坏双亲委派机制的方法:

1. 自定义类加载器

通过继承ClassLoader并覆盖loadClass方法,可以实现不使用双亲委派机制的类加载器。

public class CustomClassLoader extends ClassLoader {
    @Override
    public Class<?> loadClass(String name) throws ClassNotFoundException {
        // 不委派给父类加载器,直接加载类
        byte[] classData = getClassData(name);
        if (classData == null) {
            throw new ClassNotFoundException();
        }
        return defineClass(name, classData, 0, classData.length);
    }

    private byte[] getClassData(String className) {
        // 读取类文件的字节码
        String path = className.replace('.', '/') + ".class";
        try (InputStream is = getClass().getResourceAsStream(path)) {
            ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
            int nextValue = 0;
            while ((nextValue = is.read()) != -1) {
                byteStream.write(nextValue);
            }
            return byteStream.toByteArray();
        } catch (IOException e) {
            e.printStackTrace();
            return null;
        }
    }
}
2. 使用不同的类加载器加载不同版本的类

通过不同的类加载器加载不同版本的类库,避免类库冲突。

public class VersionClassLoader extends ClassLoader {
    private String version;

    public VersionClassLoader(String version) {
        this.version = version;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        String path = "/lib/" + version + "/" + name.replace('.', '/') + ".class";
        try (InputStream is = getClass().getResourceAsStream(path)) {
            if (is == null) {
                throw new ClassNotFoundException(name);
            }
            ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
            int nextValue = 0;
            while ((nextValue = is.read()) != -1) {
                byteStream.write(nextValue);
            }
            byte[] classData = byteStream.toByteArray();
            return defineClass(name, classData, 0, classData.length);
        } catch (IOException e) {
            throw new ClassNotFoundException(name, e);
        }
    }
}
3. 重写findClass方法

通过重写ClassLoaderfindClass方法,自定义类加载逻辑,从而绕过父类加载器。

public class CustomFindClassLoader extends ClassLoader {
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] classData = getClassData(name);
        if (classData == null) {
            throw new ClassNotFoundException();
        }
        return defineClass(name, classData, 0, classData.length);
    }

    private byte[] getClassData(String className) {
        String path = className.replace('.', '/') + ".class";
        try (InputStream is = getClass().getResourceAsStream(path)) {
            ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
            int nextValue = 0;
            while ((nextValue = is.read()) != -1) {
                byteStream.write(nextValue);
            }
            return byteStream.toByteArray();
        } catch (IOException e) {
            e.printStackTrace();
            return null;
        }
    }
}

总结

双亲委派机制通过递归委派机制确保了Java类加载的安全性、一致性和性能。但是,为了满足某些特定需求,有时需要破坏这一机制。通过自定义类加载器并覆盖loadClassfindClass方法,可以实现不依赖双亲委派机制的类加载逻辑。尽管如此,在破坏双亲委派机制时应谨慎操作,以避免潜在的类加载冲突和安全问题。

标签:委派,name,详解,双亲,JVM,classData,机制,加载
From: https://blog.csdn.net/hui_zai_/article/details/139710575

相关文章

  • 【JVM】类的⽣命周期和类加载的过程
    在Java中,类的生命周期和类加载过程是Java虚拟机(JVM)管理的核心部分。类的生命周期包括从类被加载到内存直到类被卸载的整个过程。类加载过程可以细分为多个阶段:加载、链接(包括验证、准备、解析)、初始化和使用。以下是详细的描述:类的生命周期加载(Loading)定义:将类的字节码......
  • 好用的库函数,qsort函数大详解(干货满满!)(进阶)
    前言:  小编在上一篇文章说了这一篇将要写qsort函数的模拟实现,那么废话不多说,现在开始进入今天的代码之旅喽!目录:1.qsort函数的模拟实现的逻辑和思路2.qsort函数模拟实现的代码实现3.代码展示1.qsort函数的模拟实现的逻辑和思路  读者朋友们是否还记得小编之前说过......
  • Redis常见数据类型及其常用命令详解
    文章目录一、Redis概述二、Redis常用命令1.通用命令1.1KEYS:查看符合模板的所有key1.2DEL:删除一个指定的key1.3EXISTS:判断key是否存在1.4EXPIRE:给一个key设置有效期,有效期到期时该key会被自动删除1.5TTL:查看一个key的剩余有效期1.6COPY:复制Redis数据库......
  • JMeter 响应断言详解:提升测试精度的利器
    前言在性能测试和功能测试中,响应断言是验证系统响应是否符合预期的重要手段。ApacheJMeter提供了丰富的断言功能,帮助测试工程师确保测试请求的响应数据正确、可靠。本文将详细介绍JMeter中响应断言的类型、配置方法以及最佳实践。什么是响应断言?响应断言用于验证JMeter发......
  • 精准控制:Python 输入数值范围限制详解
    前言在实际开发过程中,经常需要对用户输入的数值进行限制,以确保输入的数据在合理的范围内。这不仅能防止程序错误,还能提高用户体验。作为一名测试工程师,掌握如何在Python中限制输入数值范围是非常有用的技能。本文将详细介绍如何使用Python实现这一功能,包括基础方法和高级应用......
  • 多模态大模型:识别和处理图片与视频的技术详解
    随着人工智能和深度学习技术的快速发展,多模态大模型在识别和处理图片与视频方面展现出了强大的能力。多模态大模型能够处理多种形式的数据,包括文本、图像、视频、音频等,从而实现更智能、更全面的理解与应用。本文将详细介绍多模态大模型是如何识别和处理图片与视频的。1.......
  • MySQL约束详解:构建数据完整性基石
    目录MySQL约束1.1约束1.1数据类型1.2主键约束[`重要`]1.3自增约束1.4唯一约束1.5非空约束1.6默认值代码演示1.7外键约束[了解]思维导图最后MySQL约束MySQL作为广泛使用的开源关系型数据库管理系统,其强大的数据约束功能对于维护数据的一致性和准确性至关重......
  • Oracle 12C的闪回技术详解
    1闪回技术介绍闪回技术是oracle强大数据库备份恢复机制的一部分,在数据库发生逻辑错误的时候,闪回技术能提供快速且最小损失的恢复(多数闪回功能都能在数据库联机状态下完成)。需要注意的是,闪回技术旨在快速恢复逻辑错误,对于物理损坏或是介质丢失的错误,闪回技术就回天乏术了,还......
  • NumPy 差分、最小公倍数、最大公约数、三角函数详解
    NumPy差分离散差分意味着相邻元素之间的减法。例如,对于[1,2,3,4],离散差分将是[2-1,3-2,4-3]=[1,1,1]要找到离散差分,使用diff()函数。示例:importnumpyasnparr=np.array([10,15,25,5])newarr=np.diff(arr)print(newarr)返回:[510-20],因为15-......
  • Java数组 详解(初始化 格式 索引 地址值 遍历 …)
    数组什么是数组?数组指的是一种容器可以用来存储同种数据类型的多个值小结:数组指的是一种容器可以用来存储同种数据类型的多个值//数组容器在存储数据的时候需要结合隐式转换考虑//例如int类型的数组容器( byte short int )//例如double类型的数组容器......