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应用时提供更多的灵活性和控制力。
原文地址:https://blog.csdn.net/weixin_41247813/article/details/140610422
免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!