JVM进阶调优系列(1)类加载器原理一文讲透

news2025/1/23 7:59:28

今天开始写JVM调优系列,并发编程系列也会继续穿插连载,让各位同学闲暇之余有更多阅读选择。

起笔写第一篇,并不好写。首先要构思整个系列的大概框架,一个好的框架一定是深度上由浅入深、逻辑上有严格顺序,读者订阅跟踪是顺畅舒服的感觉。而且广度上也要尽可能的的齐全,所以第一篇应该写什么呢?

.java文件如何运行?

java对象的创建流程和内存分配,生命周期是怎样?

jvm类加载器机制剖析?

jvm垃圾收集器有几种?

工作中的GC问题如何排查解决?

jvm工作实战案例xx分析?

....

思辨比较一番,其实不管从实战开笔、还是理论基础开局都要遵从由浅入深、文章内容连贯的出发点,决定保持和并发系列写作风格,结合实际实用案例代码到知识点,让文章阅读变得简单、有趣、实用!

整个系列框架大概是围绕JVM类加载器机制、内存模型JMM、对象生命周期管理、垃圾回收机制、GC实战进行展开。

一、类加载机制是什么?

类加载机制,就是JVM进程通过类加载器classLoader将.class文件加载到内存解析、运行的过程。那.class文件如何被加载和运行的呢?

1.1 java代码是如何运行起来的?

1、首先.java文件,通过javac命令编译或者通过mvn打包变成jar、war包,java文件就变成.class文件。

2、然后运行.class文件,通过java -jar xxx来运行。那具体的某个类class文件,什么时候被加载到jvm内存中?

比如以下代码,什么时候会加载User.class文件?当执行代码要用到这个类的时候就会被加载。

在执行Demo001ClassLoader的main方法时候,发现有调用getUser()方法,而方法里有实例化User类,这时候就会去加载User.class文件。

public class Demo001ClassLoader {
    public User getUser(String userName) {
        User user = new User(userName);
        return user;
    }
    public static void main(String[] args) {
        System.out.println("类加载器机制");
        Demo001ClassLoader classLoader = new Demo001ClassLoader();
        classLoader.getUser("拉丁");
    }
}

jvm进程通过类加载器加载相关类的class文件到内存执行,这个时候就涉及要理解类加载器的机制。

二、有多少种类加载器?

2.1 启动类加载器(Bootstrap ClassLoader)

用来加载 Java 的核心类,java的核心类就是我们安装JDK的时候,包里面有个lib目录,里面的文件就是java的核心类库。

2.2 扩展类加载器(Extention ClassLoader)

扩展类加载器负责加载 JDK安装包lib 目录下还有一个ext 目录,这个ext目录下的或者被 java.ext.dirs 系统变量所指定的路径中的所有类库。

2.3 应用类加载器(Application ClassLoader)

负责在 JVM 启动时加载用户类路径上的指定类库,比如我们开发的java程序,就是由这个应用类加载器来加载。

2.4 用户自定义类加载器(User ClassLoader)

当上述 3 种类加载器不能满足需求时,我们可以继承 java.lang.ClassLoader 类,自定义一个类加载器。在自定义的累加器里如果想打破双亲委派机制,那么可以重写 loadClass 方法;如果不想打破双亲委派机制,那么只需要直接重写 findClass 方法即可。

三、具体说说双亲委派机制原理?

jvm收到一个类加载的请求,是如何安排的呢?四种类加载器,到底哪个类加载器会去加载?

jvm默认的加载机制就是双亲委派机制。这个机制,就是一个【父子层级结构关系】图。每个类加载器都有一个父加载器。

自定义类加载器的父加载器是【应用类加载器】。

应用类加载器的父加载器就是【扩展类加载器】。

扩展类加载器的父加载器就是【启动类加载器】。

双亲委派机制(实际就是父类委派)核心原理:一个类加载器收到一个类加载请求时,先委托父加载器去加载。如果父加载器还有父级,继续递归委托,请求最终到达最顶级加载器,也就是启动类加载器Bootstrap ClassLoader。

启动类加载器判断是否在自己的加载范围目录下,如果在就加载返回成功,不在的话就把加载任务下推交给下一级加载器-扩展类加载器,扩展类加载器也是类似如此。最后如果子类加载器本身也加载不到这个类就报ClassNotFoundException异常。

一句话:类加载任务先上推给父加载器,上推递归直到启动类加载器才开始尝试加载。如果启动类加载器加载不到该类,就开始下发分配给子类加载器。

再简单就是:类加载任务来了,先委派父级加载器去处理。父类加载器加载不到,自己才去加载。

四、双亲委派机制的优点是什么、缺点是什么?

4.1 双亲委派机制的优点

避免重复加载:保证每个类只被加载一次。

安全性:由于每个类只被加载一次,确保全局唯一,避免核心api被篡改。

4.2 双亲委派机制的缺点

缺点1:子类加载器可以使用父类加载过的类,但是父类加载器无法使用子类加载器加载过的类。

比如JDK有很多服务提供者接口SPI(Service provider Interface),像jdbc、JDNI接口,这些是java的核心库,都是在JDK包的lib目录下。负责加载这个目录的是启动类加载器。实现这些SPI接口的是第三方自定义包,比如MySQL的jdbc、oracle的jdbc,这种自定义的包,按理应该在自定义类加载器里加载。

按双亲委派机制,在应用程序执行到SPI接口实现方法,启动类加载器从lib目录下加载完SPI接口后,jvm发现这个接口实现方法的代码还在自定义类加载器负责范围里,这时候把启动类加载器难倒了!【我要加载一个类,但是我加载不到,而且我没有父加载器委托,更bug 的是我无法向下委托加载】。

4.3 打破双亲委派机制的方式

双亲委派机制并不是一个强制约束,而是 Java 设计者推荐给我们的类加载器的实现方式。所以为了完成某些特定操作,我们可以“打破” 这个机制。

打破双亲委派模型的方法主要包括:

1、重写 loadClass() 方法,比如我们自定义类加载器,如果要打破双亲委派机制,我们就重写loadClass()方法就可以。

2、利用线程上下文加载器。Java 应用上下文加载器默认是使用 AppClassLoader。若想要在父类加载器使用到子类加载器加载的类,可以使用 Thread.currentThread().getContextClassLoader()。

        String name = "java/sql/Date.class";
        Enumeration<URL> urls = Thread.currentThread().getContextClassLoader().getResources(name);
        while (urls.hasMoreElements()) {
            URL url = urls.nextElement();
            System.out.println(url.toString());
        }

五、Tomcat如何打破双亲委派机制?

Tomcat是一个web容器,需要部署多个应用。每个应用的依赖代码可能是不同的版本。比如A应用的fastJson是2.0版本,B应用是3.0版本,里面都有JASONArray类。但按双亲委派机制,不可以重复加载同一个类。

Tomcat 为每个 web 容器单独提供一个 WebAppClassLoader 加载器,通过提供隔离的机制,破坏双亲委派原则。

实现流程大概如下:

1、为每一个应用在容器里有单独的 WebAppClassLoader 加载器,该加载器负责只加载应用自身目录下的 class 文件,从而实现隔离。

2、如果WebAppClassLoader加载不到,才向上委派到通用的加载器 CommonClassLoader 进行加载。

今天就分享到这,说完类加载器种类、优缺点,以及如何打破双亲委派机制后,那么JVM进程通过类加载器classLoader将.class文件加载到内存的过程具体做哪些操作?解析、验证?留一个思考题给大家,下一篇文章,我们再细说。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2203488.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

免费获取的8个SVG图标库,轻松下载与复制!

SVG图标相比传统的JPG、PNG图标具有诸多优势&#xff0c;适用于各种类型的图像&#xff0c;不仅能在不同尺寸下保持清晰度&#xff0c;还具备高度压缩性和轻量特性&#xff0c;支持静态和动态效果。因此&#xff0c;SVG格式在网页设计中往往是优选。尽管如今有很多免费的图标库…

风扇PD协议取电协议芯片-ECP 5702

随着USB-C的普及&#xff0c;市面上消费者PD充电器越来越多&#xff0c;如何让小家电产品也能够支持PD协议快充呢&#xff1f;加入一颗能芯科技PD协议取电协议芯片ECP5702试试看 USB PD协议受电端诱骗协议芯片 1、概述 ECP5702是能芯科技开发的一款专门PD协议的Sink控制器。 …

【论文速看】DL最新进展20241010-扩散模型、目标检测、行人检测

目录 【扩散模型】【目标检测】【行人检测】 【扩散模型】 []Faster Diffusion: Rethinking the Role of UNet Encoder in Diffusion Models 论文链接&#xff1a;https://arxiv.org/pdf/2312.09608 代码链接&#xff1a;https://github.com/hutaiHang/Faster-Diffusion 扩散…

No.10 笔记 | PHP学习指南:PHP数组掌握

本指南为PHP开发者提供了一个全面而简洁的数组学习路径。从数组的基本概念到高级操作技巧&#xff0c;我们深入浅出地解析了PHP数组的方方面面。无论您是初学者还是寻求提升的中级开发者&#xff0c;这份指南都能帮助您更好地理解和运用PHP数组&#xff0c;提高编码效率和代码质…

java批量发送邮件:如何实现高效邮件群发?

java批量发送邮件的教程指南&#xff1f;利用Java实现邮件批发&#xff1f; 随着技术的进步&#xff0c;java批量发送邮件已经成为企业实现高效邮件群发的关键工具。AokSend将探讨如何利用java批量发送邮件技术&#xff0c;实现高效的邮件群发&#xff0c;提升营销效果。 jav…

相当炸裂!495页看漫画学Python(全彩版)通俗易懂!Git首发破万Star

今天给大家分享一份由清华大学出品的《看漫画学Python》&#xff0c;本书作者对每一幅漫画表达的准确性也进行了N遍的推敲和打磨&#xff0c;向广大读者奉献一本精品漫画Python技术书。 总共495页&#xff0c;书中结合了幽默的故事情节和实用的编程知识&#xff0c;使得学习过…

【LeetCode】动态规划—673. 最长递增子序列的个数(附完整Python/C++代码)

动态规划—673. 最长递增子序列的个数 前言题目描述基本思路1. 问题定义2. 理解问题和递推关系3. 解决方法3.1 动态规划方法3.2 优化方法 4. 进一步优化5. 小总结 代码实现PythonPython3代码实现Python 代码解释 CC代码实现C 代码解释1. 初始化&#xff1a;2. 动态规划过程&…

Basic Pentesting靶机打靶记录

一、靶机介绍 下载链接&#xff1a;https://download.vulnhub.com/basicpentesting/basic_pentesting_1.ova 二、信息收集 确认靶机ip&#xff1a;192.168.242.136 arp-scan -l 扫描端口 nmap -p- -A -sS 192.168.242.136 这里开放了21&#xff0c;22&#xff0c;80端口 扫…

美发店数字化转型:SpringBoot管理系统

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

鸿蒙开发:文件推送到沙箱路径

最近一个项目需要基于沙箱路径下的文件进行操作&#xff0c;奈何应用沙箱路径下没有。找来找去方法都是要把文件推送进去。以下是我的一些拙见&#xff0c;请各位看官老爷指点一二。 沙箱路径 沙箱路径&#xff08;Sandbox Path&#xff09;通常是指在计算机安全和软件开发中…

【大学学习-大学之路-回顾-电子计算机相关专业-学习方案-自我学习-大二学生(2)】

【大学学习-大学之路-回顾-电子&计算机相关专业-学习方案-自我学习-大二学生&#xff08;2&#xff09;】 1、前言2、总体说明1-保证课程原因1&#xff1a;原因2&#xff1a; 2-打比赛3-自我适应 - 享受大学生活 3、 保证课程1、英语课程2、专业课程3、其他课程 4、 打比赛…

数据质量指标:如何衡量数据的准确性

数据质量是任何数据驱动运营的重要组成部分。即使对于不打算将数据集出售给其他公司的企业&#xff0c;数据的质量和准确性也会极大地影响决策效率。 不幸的是&#xff0c;没有单一指标可以确保数据质量达到标准。您必须跟踪多个指标并不断关注它们。因此&#xff0c;维护数据…

高通QCS6490开发(十):合并显示多路安防摄像头

视频分析时边缘侧AI应用的一个常见场景&#xff0c;边缘侧的单个节点能够同时视频流越多&#xff0c;这不仅提高了处理效率&#xff0c;还具有显著的经济性。本文将介绍如何使用QCS6490的VPU&#xff08;视频处理单元&#xff09;来支持H264/H265的视频硬件编解码&#xff0c;并…

C语言计算GPS卫星位置

1 概述 在用GPS信号进行导航定位以与制订观测计划时&#xff0c;都必须已知GPS卫星在空间的瞬间位置。卫星位置的计算是根据卫星电文所提供的轨道参数按一定的公式计算的。本节专门讲解观测瞬间GPS卫星在地固坐标系中坐标的计算方法。 2 卫星位置的计算 1. 计算卫星运行的平…

如何做好项目管理中的需求管理?

本人任职于某科技公司项目经理&#xff0c;主要帮助客户梳理现有的业务流程&#xff0c;借助公司自主研发的低代码平台实现流程的线上化&#xff0c;业务的数字化转型。 由于项目性质特殊&#xff0c;在实施期间&#xff0c;对于总体项目需要采用传统的瀑布式开发规划整个项目…

揭秘网络流量分析的秘密 WireShark使用教程

WireShark是一个网络包分析工具。该工具主要用来捕获网络数据包&#xff0c;并自动解析网络数据包&#xff0c;为用户显示数据包详细信息&#xff0c;供用户对数据包进行分析 网络管理员 使用WireShark来检查网络问题网络安全工程师 使用WireShark来检查咨询安全相关问题开发人…

腾讯云大牛亲码“redis深度笔记”在牛客网上火了,完整 PDF 开源

前言 作为这个时代码代码的秃头人员&#xff0c;对Redis肯定是不陌生的&#xff0c;如果连Redis都没用过&#xff0c;还真不好意思出去面试&#xff0c;指不定被面试官吊打多少次。 毕竟现在互联网公司和一些创业公司都要用到Redis&#xff0c;像亚马逊、谷歌、阿里、腾讯都要…

【大模型理论篇】精简循环序列模型(minGRU/minLSTM)性能堪比Transformer以及对循环神经网络的回顾

1. 语言模型之精简RNN结构 近期关注到&#xff0c;Yoshua Bengio发布了一篇论文《Were RNNs All We Needed?》&#xff0c;提出简化版RNN&#xff08;minLSTM和minGRU&#xff09;。该工作的初始缘由&#xff1a;Transformer 在序列长度方面的扩展性限制重新引发了对可在训练期…

6款支持多平台的电脑监控软件,电脑多屏监控软件

在当今信息化办公环境中&#xff0c;监控软件已成为企业提升工作效率、管理公司资源的关键工具。随着远程办公与多设备管理的兴起&#xff0c;具备多平台兼容性和强大功能的电脑监控软件&#xff0c;能够帮助管理者随时掌握员工的工作情况、数据安全以及企业运营状态。本文将推…

【全解析】从xinput1_3.dll的作用到解决xinput1_3.dll相关问题的深度探究

在计算机系统的运行过程中&#xff0c;我们有时会遇到各种各样的文件缺失问题&#xff0c;其中xinput1_3.dll文件的缺失是比较常见的一种情况。今天这篇文章将和大家聊聊从xinput1_3.dll的作用到解决xinput1_3.dll相关问题的深度探究&#xff0c;将电脑恢复正常。 xinput1_3.dl…