Java类加载与JVM详解:从基础到双亲委托机制

        在Java开发中,理解JVM(Java虚拟机)和类加载机制是掌握高级特性的关键。本文将从JDK、JRE、JVM的关系入手,深入讲解JVM的内存结构,并详细剖析类加载的全过程,包括加载时机、流程以及核心机制——双亲委托模型


一、JDK、JRE、JVM的关系

1.1 三者的核心区别

  • JDK(Java Development Kit):Java开发工具包,包含编译器(javac)、调试器(jdb)等开发工具,是开发Java程序的必备环境。
  • JRE(Java Runtime Environment):Java运行时环境,包含JVM和Java类库,用于运行Java程序。
  • JVM(Java Virtual Machine):Java虚拟机,是JRE的核心组件,负责执行字节码(.class文件),并提供内存管理、垃圾回收等功能。

关系图


1.2 Java程序跨平台原理


二、JVM内存结构详解

        JVM(Java Virtual Machine) 是Java平台的核心组件,它提供了跨平台的 能力,使得Java程序可以在不同的操作系统上运行。JDK中的JVM负责解释和执 行Java字节码文件,同时还提供了内存管理、垃圾回收等功能,使得Java程序能 够高效、安全地运行。

JVM的内存结构是Java程序运行的基石,主要分为以下几部分:

类加载器(Class Loader):类加载器负责加载Java字节码文件(.class文件), 并将其转换为可执行的代码。它将类加载到JVM的运行时数据区域中,并解析类 的依赖关系

运行时数据区(Runtime Data Area):运行时数据区域是JVM用于存储程序运行 时数据的区域。它包括以下几个部分:

2.1 核心区域划分

区域作用特点
方法区(Method Area)存储类的元数据(如类结构、常量池、静态变量)线程共享,可能存在性能瓶颈(如频繁GC)
堆(Heap)存放对象实例和数组最大的内存区域,垃圾回收的主要场所
栈(Stack)存储局部变量、方法调用栈帧线程私有,生命周期与线程一致
本地方法栈(Native Method Stack)支持本地方法(如C/C++代码)调用与JVM栈类似,但服务对象不同
程序计数器(Program Counter)记录当前线程执行的字节码指令地址线程私有,唯一不会抛出OOM异常的区域

2.2 执行引擎与垃圾回收

  • 执行引擎::执行引擎负责执行编译后的字节码指令,将其 转换为机器码并执行。它包括解释器即时编译器(Just-In-Time Compiler, JIT)两个部分,用于提高程序的执行效率。
  • 垃圾回收器(GC)::垃圾回收器负责自动回收不再使用的对象和 释放内存空间。它通过标记-清除、复制、标记-整理等算法来进行垃圾回收
  • 本地方法接口(Native Method Interface):本地方法接口允许Java程序调用本 地方法,即使用其他语言编写的代码。

三、类加载机制详解

类加载是Java动态性的核心,JVM通过类加载器将.class文件加载到内存,并完成初始化。

JVM架构及执行流程如下:

解释执行:

        class文件内容,需要交给JVM进行解释执行,简单理解就是JVM解释一行就 执行一行代码。所以如果Java代码全是这样的运行方式的话,效率会稍低一 些。

JIT(Just In Time)即时编译:

        执行代码的另一种方式,JVM可以把Java中的 热点代码直接编译成计算机可 以运行的二进制指令,这样后续再调用这个热点代码的时候,就可以直接运 行编译好的指令,大大提高运行效率。

3.1 类加载器

类加载器可以将编译得到的 .class文件 (存储在磁盘上的物理文件)加载在 到内存中。

3.2 加载时机

当第一次使用到某个类时,该类的class文件会被加载到内存方法区。

3.3 加载过程

类加载的过程:加载、验证、准备、解析、初始化

具体加载步骤:

注意事项:

        类加载过程是按需进行的,即在首次使用类时才会触发类的加载和初始化。此 外,类加载过程是由Java虚拟机的类加载器负责完成的,不同的类加载器可能有 不同的加载策略和行为。

类加载小节:

        JVM的类加载过程包括加载、验证、准备、解析和初始化等阶段,它们共同完 成将Java类加载到内存中,并为类的静态变量分配内存、解析符号引用、执行静 态代码块等操作,最终使得类可以被正确地使用和执行。

3.4 加载器分类

JDK8类加载器可以分为以下四类:

3.5 双亲委托机制(Parent Delegation Model)

        双亲委托机制是Java类加载器的一种工作机制,通过层级加载和委托父类加载器 来加载类,确保类的唯一性、安全性和模块化。在学习这个知识点之前,大家看 下面题目。

1)问题引入

        用户自定义类java.lang.String,在测试类main方法中使用该类,思考: 类加载器到底加载是哪个类,是JDK提供的String,还是用户自定义的String?

自定义String类:
package java.lang;import java.util.Arrays;public class String {private char[] arr;public String() {System.out.println("in String() ...");arr = new char[10];}public String(char[] array) {System.out.println("in String(char[]) ...");int len = array.length;arr = new char[len];System.arraycopy(array, 0, arr, 0, len);}@Overridepublic String toString() {return "MyString: " + Arrays.toString(arr);}
}

测试类:

import java.lang.String;public class Test035_String {public static void main(String[] args) {String s1 = new String();System.out.println("s1: " + s1);}}

从运行效果可知:最终加载的类是JDK提供的java.lang.String

为什么?答案是双亲委托机制

2)双亲委托机制

        如果一个类加载器收到类加载请求,它并不会自己先去加载,而是把这个请求 委托给父类的加载器去执行,如果父类加载器还存在其父类加载器,则进一步向 上委托,最终加载请求会到达顶层的启动类加载器 Bootstrap ClassLoader 。

        如果顶层类加载器可以完成加载任务,则进行class文件类加载,加载成功后返 回。如果当前类加载器无法加载,则向下委托给子类加载器,此时子类加载器才 会尝试加载,成功则返回,失败则继续往下委托,如果所有的加载器都无法加载 该类,则会抛出ClassNotFoundException,这就是双亲委托机制。

3.6 常用方法

案例展示:

        准备一个jdbc的配置文件 jdbc.properties ,借助类加载器中方法解析,遍 历输出其配置内容。

配置文件 jdbc.properties :

 driverClass=com.mysql.jdbc.Driverurl=jdbc:mysql://localhost:3306/db01username=rootpassword=briup

测试类:

 import java.io.IOException;import java.io.InputStream;import java.util.Map.Entry;import java.util.Properties;import java.util.Set;// static ClassLoader getSystemClassLoader()    
获取系统类加载器
// InputStream getResourceAsStream(String name) 加载当前类class
文件相同目录下资源文件
public class Test036_LoadFile {public static void main(String[] args) throws IOException 
{//1.获取系统类加载器ClassLoader systemClassLoader = 
ClassLoader.getSystemClassLoader();//2.利用加载器去加载一个指定的文件// 参数:文件的路径(注意,该路径为相对路径,相对于当前类
class文件存在的目录)// 返回值:字节流InputStream is = 
systemClassLoader.getResourceAsStream("jdbc.properties");System.out.println("is: " + is);//3.实例化Properties对象,解析配置文件内容并输出Properties prop = new Properties();prop.load(is);//配置文件内容遍历Set<Entry<Object, Object>> entrySet = prop.entrySet();for (Entry<Object, Object> entry : entrySet) {String key = (String) entry.getKey();String value = (String) entry.getValue();System.out.println(key + ": " + value);}//4.关闭流is.close();}}

运行效果:

注意事项:getResourceAsStream(String path),参数path是相对路径,相对当前 测试类class文件所在的目录!


总结

        本文从JDK/JRE/JVM的关系入手,深入解析了JVM的内存结构和类加载机制,重点讲解了双亲委托模型的设计原理和作用。理解这些内容有助于优化程序性能、排查类加载冲突问题,并为后续学习反射、动态代理等高级特性打下基础。

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

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

相关文章

准备机试--图【y总版】[重要]【最短路】

常用代码模板3——搜索与图论 - AcWing 一般&#xff0c;稀疏图&#xff08;m约等于n&#xff09;:堆优化版本的dj&#xff1b;稠密图&#xff08;mn^2&#xff09;&#xff1a;朴素dj 最短路的难点在于建图【抽象出点和边】 朴素dj

Python API接口实战指南:从入门到精通

&#x1f31f; Hello&#xff0c;我是蒋星熠Jaxonic&#xff01; &#x1f308; 在浩瀚无垠的技术宇宙中&#xff0c;我是一名执着的星际旅人&#xff0c;用代码绘制探索的轨迹。 &#x1f680; 每一个算法都是我点燃的推进器&#xff0c;每一行代码都是我航行的星图。 &#x…

Spring和mybatis整合后事务拦截器TransactionInterceptor开启提交事务流程

目录一、说明二、TransactionInterceptor开启事务&#xff08;1&#xff09;、拦截方法&#xff08;2&#xff09;、开启事务绑定数据库连接&#xff08;3&#xff09;、mybatis中sql执行数据库连接获取&#xff08;4&#xff09;、事务提交和当前线程ThreadLocal清理&#xff…

05.《ARP协议基础知识探秘》

ARP协议基本介绍与实践 文章目录**ARP协议基本介绍与实践**ARP概述ARP报文类型ARP工作过程解析ARP工作原理示意图无故ARP/免费ARP实验案例**实验目标**实验环境实验步骤ARP概述 作用&#xff1a;ARP&#xff08;Address Resolution Protocol&#xff0c;地址解析协议&#xff…

互联网大厂面试:大模型应用开发岗位核心技术点解析

互联网大厂面试&#xff1a;大模型应用开发岗位核心技术点解析 第一轮&#xff1a;大模型基础与上下文工程 问题 1&#xff1a;你能简单介绍 Transformer 架构的工作原理吗&#xff1f; 小C&#xff1a;嗯&#xff0c;我理解是 Transformer 主要依赖自注意力机制&#xff08;Se…

【深度学习新浪潮】有没有什么方法可以将照片变成线描稿,比如日式漫画的那种?

一、技术原理与研究进展 1. 线描生成的核心技术路径 传统方法:基于边缘检测(如Canny算子)和形态学操作,但难以处理复杂纹理和艺术风格。 深度学习方法: 端到端生成:使用U-Net架构(如ArtLine项目)直接学习照片到线描的映射,结合自注意力机制和感知损失提升细节保留能力…

NV032NV037美光固态闪存NV043NV045

NV032NV037美光固态闪存NV043NV045在数字化浪潮席卷全球的当下&#xff0c;存储技术的每一次突破都深刻影响着从个人消费到企业级应用的各个领域。美光科技作为行业领军者&#xff0c;其NV系列固态闪存产品始终以技术创新为核心驱动力。本文将聚焦NV032、NV037、NV043、NV045四…

天硕G40工业固态硬盘破解轨道存储难题

在高铁与轨道交通高速发展的今天&#xff0c;轨道检测探伤是保障列车安全运行的核心环节。据统计&#xff0c;我国铁路总里程已突破16万公里&#xff0c;日均检测数据量超10TB。加固平板一体机作为轨道探伤领域的“移动工作站”&#xff0c;需要在跨越大江南北的极端环境中实时…

基于Velero + 阿里云 OSS的Kubernetes 集群的备份与恢复

在 Kubernetes&#xff08;K8s&#xff09;中&#xff0c;备份和恢复是保障数据安全与业务连续性的关键环节&#xff0c;主要方式包括 ETCD 备份恢复 和 Velero 备份恢复&#xff0c;两者在备份粒度、恢复影响范围、存储位置等方面存在以下差异&#xff1a; 1、ETCD 备份恢复&…

解构与重构:“真人不露相,露相非真人” 的存在论新解 —— 论 “真在” 的行为表达本质

解构与重构&#xff1a;“真人不露相&#xff0c;露相非真人” 的存在论新解 —— 论 “真在” 的行为表达本质绪论&#xff1a;传统解释的突围 —— 从 “藏才” 到 “存真”“真人不露相&#xff0c;露相非真人” 这句谚语&#xff0c;自明代《西游记》以降&#xff0c;便长期…

数据结构:哈希表、排序和查找

一、哈希算法1.将数据通过哈希算法映射成一个健值&#xff0c;存取都在同一个位置&#xff0c;实现数据的高效存储和查找&#xff0c;时间复杂度由O(n)->O(1)2.哈希碰撞&#xff1a;多个数据通过哈希算法得到的键值相同二、哈希表1.构建哈希表存放0-100之间的数据2.哈希算法…

【Java基础】Java I/O模型解析:BIO、NIO、AIO的区别与联系(Netty入门必备基础)

Java I/O模型深度解析&#xff1a;BIO、NIO、AIO的区别与联系 引言 在Java的网络编程与文件操作中&#xff0c;I/O&#xff08;输入/输出&#xff09;模型是绕不开的核心话题。从早期的BIO&#xff08;Blocking I/O&#xff09;到Java 1.4引入的NIO&#xff08;Non-blocking I/…

windows PowerToys之无界鼠标:一套键鼠控制多台设备

&#x1f4bb;简介 在多设备协作的工作场景中&#xff0c;如何实现一套键鼠控制多台设备了&#xff1f;微软推出的 PowerToys 工具集中的 Mouse Without Borders&#xff08;无界鼠标&#xff09;&#xff0c;通过软件层实现跨设备的键鼠共享与数据同步功能&#xff0c;为多台…

一道比较难的sql题,筛选出重复字段的行数

select * from 导入数据表; id city_column 1 北京,上海,广州 2 上海,上海,深圳 3 北京,杭州,北京 4 上海,广州,深圳select substring_index(khmc,,,1), * from 导入数据表 truncate table 导入数据表 select count(distinct khmc) from 导入数据表; …

【K8s】整体认识K8s之与集群外部访问--service

这一篇文章主要是对service发现新的理解 为什么要使用service服务发现&#xff1f; 首先pod的IP&#xff0c;是动态的&#xff0c;当我们重启一个pod的时候&#xff0c;它会给它分配一个新的IP&#xff0c;但是如果微服务a想要去调用微服务b&#xff0c;他是需要知道微服务b所有…

k8s(自写)

kubernetes k8s是什么&#xff1f;Kubernetes是什么&#xff1f;架构是怎么样的&#xff1f;6分钟快速入门_哔哩哔哩_bilibili kubernetes是google开源神器&#xff0c;介于应用服务和服务器之间&#xff0c;能够通过策略协调和管理多个应用服务&#xff0c;只需要一个yaml文…

实现微信小程序的UniApp相机组件:拍照、录像与双指缩放

在微信小程序开发中&#xff0c;相机功能已成为许多应用的核心组成部分。本文将介绍如何使用UniApp框架实现一个功能丰富的相机组件&#xff0c;支持拍照、录像、前后摄像头切换以及双指缩放等功能。功能概述这个相机组件具备以下核心功能&#xff1a;拍照功能&#xff1a;支持…

python pyqt5开发DoIP上位机【诊断回复的函数都是怎么调用的?】

目录 文章合集 一、底层网络接收:`_receive_loop`(触发起点) 调用时机: 核心代码: 作用: 二、数据解析:`handle_received_data`(判断是否为诊断回复) 调用时机: 核心代码(诊断回复相关部分): 作用: 三、UI显示:`add_trace_entry`(展示到界面) 调用时机: 信号…

谈物质的运动与运动的物质

运动的物质是不是物质的运动&#xff0c;如果假设是&#xff08;第一假设&#xff09;&#xff0c;那末运动的物质是物质的运动&#xff0c;而运动是物质的根本属性&#xff0c;又运动的物质是物质&#xff0c;则物质的运动是物质&#xff0c;既然运动是物质的根本属性&#xf…

【MLLM】多模态理解Ovis2.5模型架构和训练流程

note 模型架构&#xff1a;延续 Ovis 系列创新的结构化嵌入对齐设计。 Ovis2.5 由三大组件构成&#xff1a;动态分辨率 ViT 高效提取视觉特征&#xff0c;Ovis 视觉词表模块实现视觉与文本嵌入的结构对齐&#xff0c;最后由强大的 Qwen3 作为语言基座&#xff0c;处理多模态嵌…