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

JVM 类加载器有哪些?双亲委派机制的作用是什么?如何自定义类加载器?

类加载器分类

大家好,我是码哥,可以叫我靓仔,《Redis 高手心法》畅销书作者。

2477fc6cf5195ee75124cafea83daf03.jpeg

先回顾下,在 Java 中,类的初始化分为几个阶段: 加载链接(包括验证、准备和解析)和 初始化

类加载器(Class Loader)则是加载阶段中,负责将本地或网络中的指定类的二进制流,加载到 Java 虚拟机中的工具。

4647a2addca94fe2bf39c98452c19f4b.png

在进入正文前,介绍下我的《Java 面试高手心法 58 讲》专栏,帮助你在现在就业压力巨大的情况下比别人多一个杀手锏,拿下心仪 offer。内容涵盖 Java 基础、Java 高级进阶、Redis、MySQL、消息中间件、微服务架构设计等面试必考点、面试高频点。

0e4bf9e45cfaa596281aba3e4b855e2f.jpeg

丢掉你收藏的那些所谓的「面试宝典」,因为它们大多数深度不够,甚至内容还有错误,你只会看完就忘,还浪费时间。这也是为何每次面试你都回答不好,找不到好工作的原因。

3be17ba12e6104b800a20cb47b2eda70.png

引导类加载器 BootstrapClassLoader

引导类加载器 BootstrapClassLoader:引导类加载器是使用 C++ 语言实现的,嵌入在 JVM 中。用于加载 Java 中的核心类库的,不继承自 java.lang.ClassLoader,在 Java 程序中通常返回 null

一般会加载 JAVA_HOME 目录下的 /jre/lib 文件夹下的 jar 和配置。

ClassLoader loader = String.class.getClassLoader();
System.out.println(loader); // 输出 null,因为 String 是由引导类加载器加载的

扩展类加载器 ExtClassLoader

扩展类加载器主要负责加载 Java 的扩展类库,一般会加载 JAVA_HOME 目录下的 /jre/lib/ext 文件夹下的 jar。

继承自 java.lang.ClassLoader,是用户可以访问的第一个类加载器。

ClassLoader extLoader = ClassLoader.getSystemClassLoader().getParent();
System.out.println(extLoader); // 输出 sun.misc.Launcher$ExtClassLoader

应用类加载器(Application ClassLoader)

应用类加载器是应用程序中默认的类加载器,可以加载 CLASSPATH 变量指定目录下的 jar,由 sun.misc.Launcher$AppClassLoader 实现。

并且一般情况下,我们编写的 Java 应用的类,都是使用该类加载器完成加载的。

ClassLoader appLoader = ClassLoader.getSystemClassLoader();
System.out.println(appLoader); // 输出 sun.misc.Launcher$AppClassLoader

类加载器抽象类 ClassLoader

在 Java 中存在一个类加载器抽象类 ClassLoader,大多数类加载器都是通过继承这个类来实现的类加载功能。以下是 ClassLoader 类的关键部分代码:

public abstract class ClassLoader {/** 类加载器的父加载器*/private final ClassLoader parent;/*** 根据类的全限定名加载类** @param name 类名称* @return     加载的Class对象* @throws ClassNotFoundException 没有发现指定类异常*/public Class<?> loadClass(String name) throws ClassNotFoundException {// 调用loadClass方法加载类,其中设置resolve=false,表示不立即解析类return loadClass(name, false);}/*** 根据类的全限定名加载类** @param name    类名称* @param resolve 是否解析这个类,true=解析,false=不解析* @return 加载的Class对象* @throws ClassNotFoundException 没有发现指定类异常*/protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {synchronized (getClassLoadingLock(name)) {// 检查类是否已经被加载Class<?> c = findLoadedClass(name);// 如果没有加载过if (c == null) {// 如果有父类加载器,则委托给父加载器去加载// 如果没有父类加载器,则判断 Bootstrap 类加载器是否加载过if (parent != null) {c = parent.loadClass(name, false);} else {c = findBootstrapClassOrNull(name);}// 如果父类加载器都加载失败,则当前类加载器尝试自行加载if (c == null) {c = findClass(name);}}// 据 resolve 参数决定是否解析类if (resolve) {resolveClass(c);}return c;}}/*** 查找并加载指定名称的类** @param name 类名称* @return Class对象* @throws ClassNotFoundException 没有发现指定类异常*/protected Class<?> findClass(String name) throws ClassNotFoundException {//1. 根据传入的类名,到在特定目录下去寻找类文件,把字节码文件读入内存// ...//2. 调用 defineClass 将字节数组转成 Class 对象return defineClass(buf, off, len);}/*** 将一个 byte[] 转换为 Class 类的实例** @param name 类名称,如果不知道此名称,则该参数为 null* @param b    组成类数据的字节数组* @param off  类数据的起始偏移量* @param len  类数据的长度* @return Class对象* @throws ClassFormatError 类格式化异常*/protected final Class<?> defineClass(byte[] b, int off, int len) throws ClassFormatError {...}}

类中定义的常用的类加载相关的方法:

方法名称描述
getParent()返回该类加载器的父类加载器
loadClass(String name)加载指定名称的类,返回 java.lang.Class 实例
findClass(String name)查找指定名称的类,返回 java.lang.Class 实例
findLoadedClass(String name)查找已加载的指定名称的类,返回 java.lang.Class 实例
defineClass(String name, byte[] b, int off, int len)将字节数组转换为一个 Java 类,返回 java.lang.Class 实例
resolveClass(Class c)连接指定的 Java 类

双亲委派模型(Parent Delegation Model)

双亲委派模型 是类加载器的设计模式,其核心思想是:类加载请求由子类加载器向父类加载器逐层委派,直到引导类加载器。

如果父类加载器无法加载,子类加载器才会尝试加载。

如果子类加载器也无法加载该类,就会抛出一个 ClassNotFoundException 异常。

1cf1bb298bddffa9ab84a0691c61bb3f.png

双亲委派机制的作用

我们试想一下,如果不使用这种委托模式,那我们就可以随时使用自定义的 String 类来动态替代 Java 核心 API 中定义的类型,这样会存在非常大的安全隐患。

而双亲委托的方式,就可以避免这种情况,因为 String 已经在启动时就被引导类加载器 (BootstrcpClassLoader) 加载,所以用户自定义的 ClassLoader 永远也无法加载一个用户自己自定义的 String 类,除非你改变 JDK 中 ClassLoader 搜索类的默认算法。

该机制的作用如下。

  • 防止重复加载字节码文件: 将类加载请求先委托给父类,父类加载后子类就不会重复加载该类。所以,双亲委派机制可以防止对某个类重复加载;

  • 防止核心字节码文件被篡改: 一般情况下引导类加载器会先加载 JVM 核心类库,然后其它加载器才会执行,如果其它加载器要加载一个被篡改的核心字节码文件,会将该文件委托给父类加载器,当委托到引导类加载器时,加载器已经加载过该类,就不会对该类进行重复加载。而且就算能被加载,那么加载它的肯定不是相同的类加载器 (不会是引导类加载器),Java 虚拟机中只认可核心类加载器加载的核心类库,所以,双亲委派机制可以防止核心字节码文件被篡改。

  • 简化加载逻辑: 通过委派模式,每个类加载器只需要关注自己负责的那部分类加载逻辑,而不必关心其他类加载器的加载细节,简化了类加载器的实现,降低了系统的复杂度。

自定义类加载器

在某些场景下,标准的类加载器无法满足需求,例如:

  1. 热部署:在 Web 服务器中动态加载或更新类。

  2. 模块隔离:在同一个 JVM 中加载不同版本的类。

  3. 加密解密:加载经过加密的 Class 文件。

默认的类加载器只能加载指定目录下的 Jar 和 Class 文件。

如果需要加载指定位置的类文件并实现一些自定义逻辑,就需要自定义类加载器。

Chaya:如何实现自定义类加载器?

步骤

  1. 继承 java.lang.ClassLoader 类。

  2. 重写 findClass() 方法,通过字节流读取 Class 文件并转换为 Class 对象。

import java.io.*;public class MyClassLoader extends ClassLoader {@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] classData = loadClassData(name);if (classData == null) {throw new ClassNotFoundException();}return defineClass(name, classData, 0, classData.length);}private byte[] loadClassData(String name) {String fileName = name.replace('.', '/') + ".class";try (InputStream is = new FileInputStream(fileName);ByteArrayOutputStream baos = new ByteArrayOutputStream()) {int buffer;while ((buffer = is.read()) != -1) {baos.write(buffer);}return baos.toByteArray();} catch (IOException e) {e.printStackTrace();return null;}}
}

示例说明

  • findClass():从文件系统加载 Class 文件,并将其定义为 Class 对象。

  • defineClass():将字节数组转换为 JVM 可执行的 Class 对象。

为了为保证类加载器都正确实现双亲委派机制,在开发自己的类加载器时,只需要重写 findClass() 方法即可。

当然,如果不想使用双亲委派机制时,就需要重写 loadClass() 方法。

打破双亲委派模型

有时为了实现特殊功能,我们需要打破双亲委派模型,例如:

  1. 热部署框架:Tomcat、Spring Boot 使用自定义类加载器加载和卸载 Web 应用。

  2. SPI(Service Provider Interface)机制:JDBC 驱动等需要通过 线程上下文类加载器 来加载用户实现的接口。

最后(Ending)

最后,顺带给大家介绍下,我的新书《Redis 高手心法》。

f5880a0dac5050e60c8a1b1558020d53.jpeg

原价 100 元,现在只需要 50。


c4d676b6585cdb946986002a988582f8.png

作者简介

《Redis 高手心法》 作者,InfoQ 签约作者、51CTO Top 红人。拥有 10 年互联网工作经验,作为后端架构师,擅长 Redis、Tomcat、Spring、Kafka、MySQL 技术,对分布式微服务架构有深入了解。

528a2dc70a6ddd73659ae1c81c894f8f.png

点击卡片,关注我,学技术

往期推荐

面试官拷打:Redis 高可用篇章中面试最常见的 6 个问题!

Redis 7.0 深度探秘:List 数据结构原理与实战指南

重生之从零设计 MySQL 架构

重生之MySQL SQL 执行的 7 大关键步骤,解锁新技能

相关文章:

JVM 类加载器有哪些?双亲委派机制的作用是什么?如何自定义类加载器?

类加载器分类 大家好&#xff0c;我是码哥&#xff0c;可以叫我靓仔&#xff0c;《Redis 高手心法》畅销书作者。 先回顾下&#xff0c;在 Java 中&#xff0c;类的初始化分为几个阶段: 加载、链接&#xff08;包括验证、准备和解析&#xff09;和 初始化。 而 类加载器&#x…...

从基态到激发态再到里德伯态的双光子激发过程

铯原子&#xff08;Cs&#xff09;从基态6S1/2到激发态6P3/2再到里德伯态44D5/2的双光子激发过程&#xff0c; 并通过数值计算和图形化展示来研究不同失谐条件下的拉比频率、AC Stark位移差以及散射概率的变化 结果显示&#xff0c;在给定的实验参数下&#xff0c;拉比频率较低…...

Clickhouse 外部存储引擎

文章目录 外部存储引擎分类MySQL引擎PostgreSQL引擎MongoDB引擎JDBC引擎ODBC引擎Kafka引擎RabbitMQ引擎File引擎URL引擎HDFS引擎 外部存储引擎分类 引擎类型描述特点MySQL从 MySQL 数据库中读取数据用于与 MySQL 数据库共享数据&#xff0c;支持读取 MySQL 表中的数据 支持 SQ…...

eclipse怎么配置jdk路径?

在Eclipse中配置JDK路径是一个简单的步骤&#xff0c;以下是配置JDK路径的步骤&#xff1a; 打开Eclipse&#xff1a;启动Eclipse IDE。 访问首选项&#xff1a; 在Eclipse的菜单栏中&#xff0c;选择 Window > Preferences&#xff08;对于Mac OS X用户&#xff0c;选择 E…...

【前端】JavaScript 中的创建对象模式要点

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: 前端 文章目录 &#x1f4af;前言&#x1f4af;对象属性值中的引号规则&#x1f4af;对象属性换行与尾随逗号的使用&#x1f4af;工厂模式&#xff1a;灵活高效的对象创建&#x1f4af;自定义构造函数&#xff1a;通过…...

GWAS分析先做后学

大家好&#xff0c;我是邓飞。 GWAS分析是生物信息和统计学的交叉学科&#xff0c;上可以学习编程&#xff0c;下可以学习统计。对于Linux系统&#xff0c;R语言&#xff0c;作图&#xff0c;统计学&#xff0c;机器学习等方向&#xff0c;都是一个极好的入门项目。生物信息如…...

【系统设计】高可用之缓存基础

缓存的缘起 使用缓存的主要原因包括提高系统性能、降低数据库负载、提升用户体验和保证系统可用性。‌ 在计算机体系结构中&#xff0c;由于处理器和存储器的处理时间不匹配&#xff0c;在处理器和一个较大较慢的设备之间插入一个更小更快的存储设备&#xff08;如高速缓存&a…...

《Java核心技术I》volatile字段

volatile字段 有多处理器的计算机能够暂时在寄存器或本地内存缓存中保存内存值&#xff0c;其结果是&#xff0c;运行在不同处理器上的线程可能看到同一个内存位置上有不同的值。编译器可以改变指令执行的顺序以使吞吐量更大化&#xff0c;编译器不会选择可能改变代码语义的顺…...

2024运维故障记 | 12/2 网易云音乐崩了

#运维故障记# 前两天看到网易云音乐崩了的新闻&#xff0c;回想了一下&#xff0c;今年从网易云音乐到支付宝、还有微软&#xff0c;近期就发生了好几起运维届的故障。 今年来不及计数了&#xff0c;先做个记录。 明年看看运维届的大故障会发生多少&#xff0c;什么原因&…...

架构设计读后——微服务

1 微服务历史 2005年&#xff1a;Dr. Peter Rodgers提出"Micro-Web-Services"概念2011年&#xff1a;一个软件架构工作组使用"microservice"来描述一中架构模式2012年&#xff1b;这个工作组正式使用"microservice"来代表这个架构2012年&#x…...

软考高级架构-9.4.4-双机热备技术 与 服务器集群技术

一、双机热备 1、特点&#xff1a; 软硬件结合&#xff1a;系统由两台服务器&#xff08;主机和备机&#xff09;、一个共享存储&#xff08;通常为磁盘阵列柜&#xff09;、以及双机热备软件&#xff08;提供心跳检测、故障转移和资源管理功能的核心软件&#xff09;组成。 …...

聊聊前端工程化

深度解析前端工程化 ​ 近年来&#xff0c;随着前端技术的快速迭代和项目复杂度的增加&#xff0c;前端开发已经从简单的页面搭建演变为专业的工程化体系。前端工程化通过工具链、标准化和流程化手段&#xff0c;不仅提高了开发效率&#xff0c;也大幅提升了项目的可维护性和协…...

“放弃Redis Desktop Manager使用Redis Insight”:日常使用教程(Redis可视化工具)

文章目录 更新Redis Insight连接页面基础解释自动更新key汉化暂时没有找到方法&#xff0c; Redis Desktop Manager在连接上右键在数据库上右键在key上右键1、添加连接2、key过期时间 参考文章 更新 (TωT)&#xff89;~~~ β&#xff59;ё β&#xff59;ё~ 现在在维护另一…...

mmdection配置-yolo转coco

基础配置看我的mmsegmentation。 也可以参考b站 &#xff1a;https://www.bilibili.com/video/BV1xA4m1c7H8/?vd_source701421543dabde010814d3f9ea6917f6#reply248829735200 这里面最大的坑就是配置coco数据集。我一般是用yolo&#xff0c;这个yolo转coco格式很难搞定&#…...

聚合支付系统/官方个人免签系统/三方支付系统稳定安全高并发 附教程

聚合支付系统/官方个人免签系统/三方支付系统稳定安全高并发 附教程 系统采用FastAdmin框架独立全新开发&#xff0c;安全稳定,系统支持代理、商户、码商等业务逻辑。 针对最近一些JD&#xff0c;TB等业务定制&#xff0c;子账号业务逻辑API 非常详细&#xff0c;方便内置…...

力扣67. 二进制求和

给你两个二进制字符串 a 和 b &#xff0c;以二进制字符串的形式返回它们的和。 示例 1&#xff1a; 输入:a "11", b "1" 输出&#xff1a;"100" 示例 2&#xff1a; 输入&#xff1a;a "1010", b "1011" 输出&#…...

网络安全中的 SOC 是什么?

当今世界&#xff0c;网络威胁日益增多&#xff0c;确保网络安全已成为各种规模企业的首要任务。网络安全讨论中经常出现的一个术语是 SOC&#xff0c;即安全运营中心的缩写。但网络安全中的 SOC 是什么呢&#xff1f; SOC在防御网络威胁、管理安全事件和全天候监控系统方面发…...

16、鸿蒙学习——Visibility与(if...else)该如何选择

在鸿蒙中我们如果要控制一个组件的显示与隐藏可以设置组件的Visibility属性&#xff0c;也可使用&#xff08;if...else&#xff09;条件控制&#xff0c;具体我们该选择哪个&#xff1f;二者有什么区别呢&#xff1f; 1、Visibility 名称描述Hidden隐藏&#xff0c;但参与布局…...

PH热榜 | 2024-12-05

1. Oopsie 标语&#xff1a;用AI和会话回放调试Flutter和React Native应用 介绍&#xff1a;Zipy推出的Oopsie是一款你唯一需要的AI赋能移动端调试工具&#xff0c;它能提供▶️会话回放、&#x1f916;错误监控、&#x1f4a1;AI生成的概要分析&#xff0c;以及&#x1f525…...

Qt Chart 模块化封装曲线图

一 版本说明 二 完成示例 此文章包含:曲线轴设置,曲线切换,单条曲线显示,坐标轴。。。 三 曲线图UI创建 在UI界面拖放一个QWidget,然后在 Widget里面放一个 graphicsView 四 代码介绍 1 头文件 #include <QString> #include <QTimer> #include <QMessa…...

Prodigal基因预测工具:3天快速掌握原核生物基因发现终极指南

Prodigal基因预测工具&#xff1a;3天快速掌握原核生物基因发现终极指南 【免费下载链接】Prodigal Prodigal Gene Prediction Software 项目地址: https://gitcode.com/gh_mirrors/pr/Prodigal 你是否正在寻找一款快速、准确的原核生物基因预测工具&#xff1f;Prodiga…...

光猫拨号下,如何把二级路由器变成‘透明网桥’?一个设置让NAS、打印机全屋可见

光猫拨号下的家庭网络优化&#xff1a;二级路由器透明化实战指南 家里NAS里的电影在客厅电视上死活刷不出来&#xff1f;书房电脑找不到卧室的无线打印机&#xff1f;这些问题往往源于家庭网络中多台路由器形成的"局域网套娃"。本文将手把手教你如何将二级路由器转化…...

嵌入式核心板选型指南:从单核到四核的精准配置与实战优化

1. 项目概述&#xff1a;从“固定套餐”到“自助餐”的嵌入式核心板选型变革最近在规划一个工业HMI项目&#xff0c;主控选型时又翻开了飞凌嵌入式的产品手册。看到AM62x系列核心板配置新增了单核、双核、四核的选项&#xff0c;第一反应是&#xff1a;这路子对了。在嵌入式开发…...

别再只问哪个大模型更强了,2026年真正决定AI Agent上限的,是向量引擎

别再只问哪个大模型更强了&#xff0c;2026年真正决定AI Agent上限的&#xff0c;是向量引擎 这两年做AI的人&#xff0c;最容易掉进一个坑。 每天盯着模型榜单看。 今天这个模型会写代码了。 明天那个模型会看视频了。 后天又有一个模型说自己推理能力更强了。 看久了以后&…...

从攻到防:手把手在Kali Linux上搭建ARP欺骗实验环境(含Wireshark分析)

构建安全的本地网络实验室&#xff1a;Kali Linux下ARP欺骗攻防实战指南 在网络安全领域&#xff0c;理解攻击原理是构建有效防御的第一步。ARP欺骗作为一种经典的中间人攻击技术&#xff0c;常被用于网络渗透测试中。本文将带你从零开始搭建一个完全隔离的虚拟实验环境&#x…...

2026年玉米膨化机市场:谁是真正的行业领航者?

面对快速发展的休闲食品市场&#xff0c;如何在竞争激烈的玉米膨化机市场里抢占先机&#xff1f;随着消费者对健康食品需求的高涨&#xff0c;五谷杂粮膨化食品逐渐成为市场上的一股热潮。本篇将深度解析2026年玉米膨化机行业的趋势、选购要点&#xff0c;并对比测评几个行业知…...

Node.js框架深度解析:从Express到Nest.js,如何选择最适合你的Web开发框架?

1. 项目概述&#xff1a;为什么Node.js框架值得你花时间研究&#xff1f;如果你是一名Web开发者&#xff0c;或者正在向这个方向转型&#xff0c;那么“Node.js框架”这个词组对你来说一定不陌生。但面对市面上林林总总的框架&#xff0c;从Express、Koa到Nest.js、Fastify&…...

Google I/O 2026 推出 Antigravity SDK:本地构建 AI Agent,灵活定制功能

Antigravity SDK 登场当开发者需要将 AI 能力嵌入自有应用时&#xff0c;常见做法是通过 API 调用远程 Agent 服务&#xff0c;但这种方式存在延迟高、定制性差、依赖网络等问题。据悉&#xff0c;Google 在 I/O 2026 大会上给出了另一种解法 ---- Antigravity SDK&#xff0c;…...

为什么92%的DeepSeek AWS部署失败?资深架构师拆解3大隐性成本陷阱与4步合规加固法

更多请点击&#xff1a; https://codechina.net 第一章&#xff1a;DeepSeek AWS部署教程 在AWS云平台上部署DeepSeek系列大语言模型&#xff08;如DeepSeek-V2、DeepSeek-Coder&#xff09;需兼顾计算性能、存储效率与网络低延迟。推荐使用g5.12xlarge或p4d.24xlarge实例类型…...

用NE555和立创EDA做个会‘叮咚’的门铃:从原理图到PCB打板的完整DIY记录

从零打造NE555叮咚门铃&#xff1a;立创EDA全流程实战指南 当电子爱好者第一次尝试将电路图转化为实物时&#xff0c;往往会面临软件操作、元件选型和生产对接的多重挑战。本文将以经典NE555叮咚门铃为例&#xff0c;手把手演示如何用立创EDA完成从原理图设计到PCB打板的完整流…...