【JVM】详解双亲委派机制

2024-06-18 13:04

本文主要是介绍【JVM】详解双亲委派机制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

双亲委派机制是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 {@Overridepublic 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;}@Overrideprotected 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 {@Overrideprotected 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方法,可以实现不依赖双亲委派机制的类加载逻辑。尽管如此,在破坏双亲委派机制时应谨慎操作,以避免潜在的类加载冲突和安全问题。

这篇关于【JVM】详解双亲委派机制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072066

相关文章

Debezium 与 Apache Kafka 的集成方式步骤详解

《Debezium与ApacheKafka的集成方式步骤详解》本文详细介绍了如何将Debezium与ApacheKafka集成,包括集成概述、步骤、注意事项等,通过KafkaConnect,D... 目录一、集成概述二、集成步骤1. 准备 Kafka 环境2. 配置 Kafka Connect3. 安装 D

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为