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

【从零开始学习JVM | 第四篇】类加载器和双亲委派机制(高频面试题)

前言:

   双亲委派机制对于面试这块来说非常重要,在实际开发中也是经常遇见需要打破双亲委派的需求,今天我们一起来探索一下什么是双亲委派机制,在此之前我们先介绍一下类的加载器。

目录

​编辑

前言:

类加载器

1. 启动类加载器(Bootstrap Class Loader)

2. 扩展类加载器(Extension Class Loader)

3. 应用类加载器(Application Class Loader)

4. 自定义类加载器(Custom Class Loader)

双亲委派机制

1. 基本概念

(1)避免类重复加载

(2)保护核心 API

(3)实现类的隔离与模块化

工作流程

双亲委派机制源码解读:

ClassLoader 的四个核心方法

核心问题

总结


类加载器

由于在jdk8以及之前没有模块化这个概念,所以jkd8之前和jdk9之后加载器有些变化,在这里介绍的是jdk8及之前的加载器。

在 Java 中,类加载器(Class Loader)负责将类的字节码(.class 文件)加载到 JVM 中。根据层级和职责,主要分为以下几类:

1. 启动类加载器(Bootstrap Class Loader)

  • 职责:加载 JVM 核心类库(如 java.lang.*java.util.*),路径通常为 $JAVA_HOME/jre/lib
  • 特点
    • 由 JVM 底层实现(C++ 编写),在 Java 代码中无法直接引用。
    • 加载最基础的类,确保 JVM 运行环境。

2. 扩展类加载器(Extension Class Loader)

  • 职责:加载 Java 扩展库(如 javax.* 包),路径为 $JAVA_HOME/jre/lib/ext
  • 特点
    • 是 java.lang.ClassLoader 的子类(Java 实现)。
    • 负责加载标准库之外的扩展功能。

3. 应用类加载器(Application Class Loader)

  • 职责:加载用户类路径(classpath)下的类,即开发者编写的代码。
  • 特点
    • 是 ClassLoader 的子类,也称为系统类加载器。
    • 是 ClassLoader.getSystemClassLoader() 的返回值。

4. 自定义类加载器(Custom Class Loader)

  • 职责:通过继承 java.lang.ClassLoader 实现,用于加载特殊来源的类(如网络、加密文件)。
  • 常见场景
    • 热部署(动态加载类)。
    • 加载自定义格式的字节码(如加密类)。
    • 实现类的隔离(如 OSGi 框架)。

注意点:不能认为一个加载器与其父类加载器的关系是继承,虽然有“父”。

需要注意的是如果我们尝试获取扩展类加载器的parent对象,得到的结果是null。并不是说其父类不是启动类加载器,只是因为启动类加载器属于JVM的一部分,使用C++实现,无法被获取到 

双亲委派机制

1. 基本概念

双亲委派指的是:当一个类加载器收到类加载请求时,它首先会将请求委派给父类加载器去尝试加载,而非立即自己加载。只有当父类加载器无法加载该类时,子类加载器才会尝试自己加载。

设计目的

(1)避免类重复加载

若多个类加载器都尝试加载同一个类,会导致内存中存在多个相同类的实例,破坏类型唯一性。双亲委派确保类只被加载一次。

(2)保护核心 API

例如,用户自定义的 java.lang.String 类不会被加载,因为 String 由启动类加载器加载。这防止恶意或错误代码覆盖 JVM 核心类,保障系统安全。

(3)实现类的隔离与模块化

不同类加载器可加载同名但不同路径的类(如插件系统),实现隔离性。例如,Web 容器(如 Tomcat)使用自定义类加载器实现多个应用的隔离。

核心原则

  • 向上委派:先让父类加载器尝试加载。
  • 向下查找:父类无法加载时,子类再尝试。

工作流程

当类加载器(如 AppClassLoader)收到类加载请求时:

  1. 检查缓存:首先检查该类是否已被加载。
  2. 委派父类:若未加载,则将请求委派给父类加载器(如 ExtClassLoader)。
  3. 递归委派:父类加载器重复步骤 1 和 2,直到到达启动类加载器。
  4. 尝试加载
    • 启动类加载器尝试加载该类,若成功则返回 Class 对象。
    • 若失败,则由扩展类加载器尝试,依此类推,直到子类加载器(如 AppClassLoader)。
  5. 抛出异常:若所有类加载器都无法加载,则抛出 ClassNotFoundException

双亲委派机制源码解读:

ClassLoader 的四个核心方法

  • loadClass:负责整体调度,按层级查找类。
  • findClass:具体去哪里找类(子类必须实现)。
  • defineClass:把类文件的二进制内容变成 JVM 能懂的 Class 对象。
  • resolveClass:解析类的内部结构,确保类能正常工作。

在加载类的时候会调用它,第一个参数是加载的类名,第二个参数是否需要解析类

protected Class<?> loadClass(String name, boolean resolve)throws ClassNotFoundException{synchronized (getClassLoadingLock(name)) {// First, check if the class has already been loadedClass<?> c = findLoadedClass(name);if (c == null) {long t0 = System.nanoTime();try {if (parent != null) {c = parent.loadClass(name, false);} else {c = findBootstrapClassOrNull(name);}} catch (ClassNotFoundException e) {// ClassNotFoundException thrown if class not found// from the non-null parent class loader}if (c == null) {// If still not found, then invoke findClass in order// to find the class.long t1 = System.nanoTime();c = findClass(name);// this is the defining class loader; record the statsPerfCounter.getParentDelegationTime().addTime(t1 - t0);PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);PerfCounter.getFindClasses().increment();}}if (resolve) {resolveClass(c);}return c;}}

代码流程:

  1. 获取加载的类,如果为空说明没有加载。
  2. 进入第一个if,判断有无父类加载器,有就调用父类加载器加载。
  3. 没有父类加载器说明需要用启动类加载器加载。
  4. 第二个if表示父类加载器加载失败,则调用此时的子类加载器加载。

我们来看看findclass方法的源码

protected Class<?> findClass(String name) throws ClassNotFoundException {int index = name.indexOf(';');String cookie = "";if(index != -1) {cookie = name.substring(index, name.length());name = name.substring(0, index);}// check loaded JAR filestry {return super.findClass(name);} catch (ClassNotFoundException e) {}// Otherwise, try loading the class from the code base URL// 4668479: Option to turn off codebase lookup in AppletClassLoader// during resource requests. [stanley.ho]if (codebaseLookup == false)throw new ClassNotFoundException(name);//      final String path = name.replace('.', '/').concat(".class").concat(cookie);String encodedName = ParseUtil.encodePath(name.replace('.', '/'), false);final String path = (new StringBuffer(encodedName)).append(".class").append(cookie).toString();try {byte[] b = AccessController.doPrivileged(new PrivilegedExceptionAction<byte[]>() {public byte[] run() throws IOException {try {URL finalURL = new URL(base, path);// Make sure the codebase won't be modifiedif (base.getProtocol().equals(finalURL.getProtocol()) &&base.getHost().equals(finalURL.getHost()) &&base.getPort() == finalURL.getPort()) {return getBytes(finalURL);}else {return null;}} catch (Exception e) {return null;}}}, acc);if (b != null) {return defineClass(name, b, 0, b.length, codesource);} else {throw new ClassNotFoundException(name);}} catch (PrivilegedActionException e) {throw new ClassNotFoundException(name, e.getException());}}

核心问题

什么是双亲委派机制?

1.在一个类加载器去加载一个类的时候,会自下而上去查找这个类有没有被加载过,如果都没有加载,就会自上而下去加载这个类,如果都无法加载就会抛出异常。

2.应用类加载器的父类是扩展类加载器,扩展类加载器的父类是启动类加载器。

3.双亲委派机制的核心是第一个确保类不会被重复加载,第二个避免恶意代码替换JDK的核心类库,保证核心类库的安全和完整。

总结

     双亲委派机制是 Java 安全模型的基石,它通过层级委派和缓存机制确保类的唯一性和安全性。理解该机制有助于排查类加载异常(如 NoClassDefFoundError),并设计出更健壮的框架和应用。

     我在学到defineClass的时候有一点疑问,在源代码执行过程中就已经编译成为了class文件,为什么还需要defineClass后来我发现,编译后的是class文件是二进制形式的,而defineClass是把它转换为JVM能识别的class对象。虽然都是class但是内容形式并不一样。

这也体现了写博客的好处,方便查漏补缺,感谢你们的阅读,你的阅读和点赞是我最大的动力。

相关文章:

  • 【从零学习JVM|第三篇】类的生命周期(高频面试题)
  • 【JVM】- 内存结构
  • 八股文——JVM
  • 电子电气架构 --- E/E架构战略
  • 【RAG排序】rag排序代码示例-简单版
  • 医疗风险预测AI模型:机器学习与深度学习方法的深度分析与实践
  • Linux应用开发之网络套接字编程(实例篇)
  • 中医有效性探讨
  • 安卓贝利自动点击器高级版下载安装教程
  • 第二届计算机视觉、机器人与自动化工程国际学术会议(CRAE 2025)
  • Easy Rules规则引擎:轻量级Java规则处理实践指南
  • Vue3 + TypeScript + Element Plus 设置表单中日期控件的宽度
  • Linux 性能利器:详解 `top` 命令的使用与输出信息解析
  • 信号的诞生:Linux进程信号的启示与奥秘
  • 【AI News | 20250609】每日AI进展
  • 第二十三课:手搓随机森林
  • aurora与pcie的数据高速传输
  • Kafka入门-Broker以及文件存储机制
  • OCR、图像分类与目标检测
  • 浏览器指纹科普 | Do Not Track 是什么?
  • phpwind 做的网站/seo排名优化怎样
  • 网站建设创业计划书模板范文/何鹏seo
  • 网站用什么技术实现/企业网站模板源码
  • 营销网站的建设流程/沈阳seo关键词排名
  • 湖南做网站的公司排名/百度网盘网页登录入口
  • wordpress文章显示标签/电脑优化大师哪个好