当前位置: 首页 > news >正文

java-双亲委派机制

Java虚拟机(JVM)中的类加载器(Class Loader)负责将类(.class文件)加载到JVM中,以便Java程序能够使用这些类。在JVM中,类加载器被组织成一种层次结构关系,这种层次结构关系中的类加载器遵循一种被称为“双亲委派模型”(Parent-First Delegation Model)的加载机制。以下是关于双亲委派机制的详细解释。
### 双亲委派模型简介
双亲委派模型要求除了顶层的启动类加载器(Bootstrap Class Loader)外,其余的类加载器都应当有自己的父类加载器。类加载器在尝试自己加载类之前,首先委托给父类加载器进行加载,只有当父类加载器加载失败(即在它的搜索范围内没有找到所需的类)时,子类加载器才会尝试自己加载该类。
### 工作原理
#### 1. 类加载器层次结构
在Java中,类加载器大致可以分为以下几种:
- **启动类加载器(Bootstrap Class Loader)**:这是JVM中最顶层的类加载器,负责加载Java标准库中的类,如`rt.jar`中的类。它是用原生代码(如C/C++)实现的,是JVM的一部分。
- **扩展类加载器(Extension Class Loader)**:负责加载Java的扩展库(`jre/lib/ext`目录或者由系统属性`java.ext.dirs`指定的目录下的类库)。
- **应用程序类加载器(Application Class Loader)**:负责加载用户类路径(ClassPath)上的所有类库。
这些类加载器之间的关系是:启动类加载器没有父类加载器,扩展类加载器的父类加载器是启动类加载器,应用程序类加载器的父类加载器是扩展类加载器。
#### 2. 加载过程
当一个类需要被加载时,以下过程会被执行:
1. **检查缓存**:首先检查该类是否已经被加载过,如果有就直接返回,否则继续下一步。
2. **委派给父类加载器**:将类加载请求委派给父类加载器,父类加载器重复步骤1和2,直到达到启动类加载器。
3. **尝试加载**:如果父类加载器无法完成加载,则当前类加载器会尝试自己加载该类。
4. **定义类**:加载完成后,将类数据存入方法区(Method Area),并在Java堆(Heap)中创建一个对应的`java.lang.Class`对象用于封装类数据。
#### 3. 优点
- **避免类的重复加载**:当父类加载器已经加载了该类时,子类加载器无需再次加载,确保了同一个类在JVM中是唯一的。
- **保护程序安全**:通过双亲委派模型,Java核心API中定义的类型不会被随意替换,防止了核心API被篡改的风险。
### 特殊情况
尽管双亲委派模型是Java推荐使用的类加载器实现方式,但在某些情况下,它可能不适用或者需要被打破:
- **SPI(Service Provider Interface)**:JDBC、JNDI、JAXP等规范,它们允许用户通过接口来扩展服务,而这些接口的实现类可能是由不同的厂商提供的,因此需要由不同的类加载器来加载。
- **热部署**:在运行时更新应用程序的某些部分,而不需要重启整个应用程序,这就需要自定义类加载器来绕过双亲委派机制。
### 自定义类加载器
如果需要自定义类加载器,通常会有以下两种方式:
- **继承`java.lang.ClassLoader`**:覆盖`findClass`方法,遵循双亲委派模型。
- **继承`java.lang.ClassLoader`并覆盖`loadClass`方法**:完全打破双亲委派模型,自定义加载逻辑。
### 安全性考虑
双亲委派模型在安全性方面起到了重要作用,它确保了Java核心库的类型安全。但是,也有潜在的安全问题,比如类加载器可能会加载恶意的类文件。因此,以下措施可以增强安全性:
- **代码签名**:对类文件进行数字签名,类加载器在加载类之前验证签名。
- **访问控制**:限制类加载器加载特定路径下的类文件。
### 总结
双亲委派模型是Java虚拟机中类加载器的一种重要机制,它通过委派的方式确保了类的唯一性和安全性。尽管在某些特殊情况下需要打破这一模型,但双亲委派模型仍然是Java平台推荐和广泛使用的类加载方式。了解双亲委派模型的工作原理和实现方式,对于Java开发人员来说是非常重要的,它不仅有助于理解Java类的加载过程,还能在开发复杂Java应用时提供更多的灵活性和控制力。
 

http://www.lryc.cn/news/405835.html

相关文章:

  • 【C++】set的使用
  • React 18【实用教程】(2024最新版)
  • Perl语言入门学习指南
  • 《Java8函数式编程》学习笔记汇总
  • C语言之封装,继承,多态
  • GO内存分配详解
  • 每日Attention学习12——Exterior Contextual-Relation Module
  • 为什么现在电销公司这么难?
  • 每天一个数据分析题(四百四十二)- 标签与指标
  • [论文笔记] pai-megatron-patch Qwen2-72B/7B/1.5B 长文本探路
  • 【SpringCloud】微服务远程调用OpenFeign
  • MySQL零散拾遗(四)
  • 大语言模型-检索测评指标
  • Zookeeper集群中节点之间数据是如何同步的
  • HTTPServer改进思路2(mudou库核心思想融入)
  • Kubernetes Secret 详解
  • docker笔记4-部署
  • 有监督学习基础
  • 揭开 AI 绘画提示词的神秘密码!
  • macOS 10.15中屏蔽Microsoft Edge浏览器的更新提示
  • Qt 实战(3)数据类型 | 3.2、QVariant
  • Docker中安装的postgresql14在启用vector扩展的时候,找不到该扩展的控制文件。
  • JS防抖和节流
  • OpenWrt 为软件包和docker空间扩容
  • 重要的工作任务,怎么在电脑桌面设置倒计时?
  • Failed to build get_cli:get:的解决方案
  • 短视频矩阵源码技术分享
  • 轮播图自定义内容
  • 大数据-44 Redis 慢查询日志 监视器 慢查询测试学习
  • Istio_01_Istio初识