JVM类的声明周期
文章目录
- 版权声明
- 生命周期概述
- 加载阶段
- 查看内存中的对象
- 连接阶段
- 连接阶段之验证
- 连接阶段之准备
- 连接阶段之解析
- 初始化阶段
- 练习题目一
- 练习题目二
- 练习题目三
- 练习题目四
- 使用阶段
- 卸载阶段
- 总结
版权声明
- 本博客的内容基于我个人学习黑马程序员课程的学习笔记整理而成。我特此声明,所有版权属于黑马程序员或相关权利人所有。本博客的目的仅为个人学习和交流之用,并非商业用途。
- 我在整理学习笔记的过程中尽力确保准确性,但无法保证内容的完整性和时效性。本博客的内容可能会随着时间的推移而过时或需要更新。
- 若您是黑马程序员或相关权利人,如有任何侵犯版权的地方,请您及时联系我,我将立即予以删除或进行必要的修改。
- 对于其他读者,请在阅读本博客内容时保持遵守相关法律法规和道德准则,谨慎参考,并自行承担因此产生的风险和责任。本博客中的部分观点和意见仅代表我个人,不代表黑马程序员的立场。
生命周期概述
Java类的生命周期包括以下阶段:
-
加载(Loading):当Java程序需要使用某个类时,JVM会检查该类是否已经被加载,如果没有加载,JVM会从磁盘中读取该类的字节码文件并创建一个Class对象,然后将该Class对象存放在方法区中。
-
验证(Verification):在加载类的过程中,JVM会对该类的字节码进行验证,以确保它符合Java虚拟机规范,不会对虚拟机造成安全上的威胁。
-
准备(Preparation):在准备阶段,JVM会为类的静态变量分配内存并设置默认值(0或null),并将这些变量存放在方法区中。
-
解析(Resolution):在解析阶段,JVM会将类中的符号引用转换为直接引用,以便于JVM能够快速访问类中的方法和变量。
-
初始化(Initialization):在初始化阶段,JVM会执行类的初始化代码,包括静态变量赋值和静态代码块的执行。如果该类有父类,JVM会先初始化父类。
-
使用(Using):在使用阶段,JVM会调用类中的方法和访问类中的变量。
-
卸载(Unloading):当JVM确定某个类实例已经不再被使用时,会将该类的Class对象从方法区中移除,这个过程称为卸载。
加载阶段
- 加载(Loading)阶段第一步是类加载器根据类的全限定名通过不同的渠道以二进制流的方式获取字节码信息。程序员可以使用Java代码拓展的不同的渠道。
- 类加载器在加载完类之后,Java虚拟机会将字节码中的信息保存到方法区中
- 类加载器在加载完类之后,Java虚拟机会将字节码中的信息保存到内存的方法区中
- 生成一个InstanceKlass对象,保存类的所有信息,里边还包含实现特定功能比如多态的信息
- 生成一个InstanceKlass对象,保存类的所有信息,里边还包含实现特定功能比如多态的信息
- 同时,Java虚拟机还会在堆中生成一份与方法区中数据类似的java.lang.Class对象
- 作用:在Java代码中去获取类的信息以及存储静态字段的数据(JDK8及之后)
- 作用:在Java代码中去获取类的信息以及存储静态字段的数据(JDK8及之后)
- 对于开发者来说,只需要访问堆中的Class对象而不需要访问方法区中所有信息
查看内存中的对象
- 推荐使用 JDK自带的hsdb工具查看Java虚拟机内存信息。工具位于JDK安装目录下lib文件夹中的sa-jdi.jar中。
- 启动命令:
java -cp sa-jdi.jar sun.jvm.hotspot.HSDB
连接阶段
-
验证(Verification):在加载类的过程中,JVM会对该类的字节码进行验证,以确保它符合Java虚拟机规范,不会对虚拟机造成安全上的威胁。
-
准备(Preparation):在准备阶段,JVM会为类的静态变量分配内存并设置默认值(0或null),并将这些变量存放在方法区中。
-
解析(Resolution):在解析阶段,JVM会将类中的符号引用转换为直接引用,以便于JVM能够快速访问类中的方法和变量。
连接阶段之验证
-
连接(Linking)阶段的第一个环节是验证,验证的主要目的是检测Java字节码文件是否遵守了《Java虚拟机规
范》中的约束。这个阶段一般不需要程序员参与。 -
主要包含如下四部分,具体详见《Java虚拟机规范》:
- 文件格式验证,比如文件是否以0xCAFEBABE开头,主次版本号是否满足当前Java虚拟机版本要求
- 元信息验证,例如类必须有父类(super不能为空)
- 验证程序执行指令的语义,比如方法内的指令执行中跳转到不正确的位置
- 符号引用验证,例如是否访问了其他类中private的方法等。
- 文件格式验证,比如文件是否以0xCAFEBABE开头,主次版本号是否满足当前Java虚拟机版本要求
- 版本号的检测:Hotspot JDK8中虚拟机源码对版本号检测
- 主版本号不能高于运行环境主版本号,如果主版本号相等,副版本号也不能超过。
连接阶段之准备
- 准备阶段为静态变量(static)分配内存并设置初始值
- 准备阶段只会给静态变量赋初始值,而每一种基本数据类型和引用数据类型都有其初始值。
- final修饰的基本数据类型的静态变量,准备阶段直接会将代码中的值进行赋值。
连接阶段之解析
- 解析阶段主要是将常量池中的符号引用替换为直接引用。
- 符号引用就是在字节码文件中使用编号来访问常量池中的内容。
- 直接引用不在使用编号,而是使用内存中地址进行访问具体的数据。
初始化阶段
-
初始化阶段会执行静态代码块中的代码,并为静态变量赋值
-
初始化阶段会执行字节码文件中clinit部分的字节码指令
-
clinit方法中的执行顺序与Java中编写的顺序是一致的
-
以下几种方式会导致类的初始化
- 访问一个类的静态变量或者静态方法,注意变量是final修饰的并且等号右边是常量不会触发初始化。
public class Demo1 {public static void main(String[] args) {int i = Demo2.i;System.out.println(i);} }class Demo2{static {System.out.println("初始化了...");}public static final int i = 0; }
- 调用Class.forName(String className)。
public class Demo3 {public static void main(String[] args) throws ClassNotFoundException {Class<?> clazz = Class.forName("init.ways.Demo4");} }class Demo4{static {System.out.println("初始化了...");} }
- new一个该类的对象时。
- 执行Main方法的当前类。
public class Demo5 {static {System.out.println("Demo5初始化了...");}public static void main(String[] args) throws ClassNotFoundException {new Demo6();} }class Demo6{static {System.out.println("Demo6初始化了...");} }
-
添加-XX:+TraceClassLoading 参数可以打印出加载并初始化的类
练习题目一
- 请给出运行结果,并说明原因
public class Test1 {public static void main(String[] args) {System.out.println("A");new Test1();new Test1();}public Test1(){System.out.println("B");}{System.out.println("C");}static {System.out.println("D");}
}
代码的执行结果为:D
A
C
B
C
B原因是:1. 静态代码块 `static {...}` 会在类加载时执行,因此会先输出 "D"。
2. `main` 方法中先输出 "A"。
3. 接着创建了两个 `Test1` 对象,因此会分别调用两次构造方法 `public Test1() {...}`。
4. 在构造方法之前,非静态代码块 `{...}` 会先执行,因此会先输出 "C"。
5. 每次创建 `Test1` 对象时,都会执行一次非静态代码块和构造方法,因此会输出两次 "C" 和两次 "B"。
- 详细分析:
- 局部代码块(也可以叫做构造代码块) 每次创建对象,调用构造器之前,都会执行该代码块中的代码
练习题目二
-
clinit指令在特定情况下不会出现,比如:如下几种情况是不会进行初始化指令执行的
- 无静态代码块且无静态变量赋值语句。
- 有静态变量的声明,但是没有赋值语句。
- 静态变量的定义使用final关键字,这类变量会在准备阶段直接进行初始化
-
直接访问父类的静态变量,不会触发子类的初始化
-
子类的初始化clinit调用之前,会先调用父类的clinit初始化方法
-
如果把new B02()去掉,结果如下
练习题目三
- 数组的创建不会导致数组中元素的类进行初始化
public class Test2 {public static void main(String[] args) {Test2_A[] arr = new Test2_A[10];} }class Test2_A {static {System.out.println("Test2 A的静态代码块运行");} }
- 运行的结果是什么都没输出
- 解释
- 创建一个类的数组不会导致该类被初始化,没有创建 Test2_A 类型的实例对象,因此该类不会被加载。
- 静态代码块是在类加载时进行初始化的,具体来说,当 JVM 加载一个类时,会先加载该类的父类(如果有父类的话),然后再加载该类本身。在加载类的过程中,JVM 会执行该类的静态代码块,以完成静态成员变量的初始化和其他一些静态操作。
练习题目四
- final修饰的变量如果赋值的内容需要执行指令才能得出结果,会执行clinit方法进行初始化
public class Test4 {public static void main(String[] args) {System.out.println(Test4_A.a);}
}class Test4_A {public static final int a = Integer.valueOf(1);static {System.out.println("Test3 A的静态代码块运行");}
}
- 运行结果为:
Test3 A的静态代码块运行
1
public class Test4 {public static void main(String[] args) {System.out.println(Test4_A.a);}
}class Test4_A {public static final int a =1;static {System.out.println("Test3 A的静态代码块运行");}
}
- 运行结果为:
1
-
如果一个类中定义了静态成员变量,并且这些成员变量都是编译期常量(比如使用 final 关键字修饰的常量),那么在访问这些常量时,编译器会直接将常量的值嵌入到字节码中,而不是在运行时动态计算。这个过程被称为编译期常量折叠(Compile-Time Constant Folding)。
-
在这种情况下,如果静态代码块中的代码并没有涉及到这些常量,那么在访问这些常量时,并不会触发类的初始化,也就不会执行静态代码块中的代码。因此,如果一个类中定义了静态成员变量,并且这些成员变量都是编译期常量,并且静态代码块中的代码并没有涉及到这些常量,那么在访问这些常量时,不会触发静态代码块的执行。
使用阶段
- 使用阶段是Java类生命周期中最重要的阶段,也是Java程序运行的核心阶段。在使用阶段中,Java类可以被创建、实例化、调用方法、访问变量等。
-
创建对象:在使用阶段中,可以通过关键字new创建一个类的对象
String str = new String("Hello World!");
-
实例化:在使用阶段中,可以通过构造方法来实例化一个类的对象。
Date date = new Date();
-
调用方法:在使用阶段中,可以通过对象来调用类中的方法
int length = str.length();
-
访问变量:在使用阶段中,可以通过对象来访问类中的变量
double pi = Math.PI;
卸载阶段
-
卸载阶段是Java类生命周期中的最后一个阶段。在卸载阶段中,Java虚拟机会卸载不再需要的类和类加载器,从而释放内存空间。
-
主要作用是清理内存,提高程序性能。
-
一个类被卸载的条件是它的所有实例都已经被销毁,同时该类的类对象和类加载器也都已经被销毁。
- 当一个类被卸载时,Java虚拟机会先卸载该类的所有实例,然后卸载该类的类对象和类加载器。如果该类有子类,子类也会被卸载。如果父类和子类都没有被卸载,那么子类的类加载器也不会被卸载。
-
Java虚拟机在卸载类时,会调用该类的finalize()方法,该方法可以被子类重写以完成一些清理工作。在finalize()方法中,可以关闭文件、释放资源等操作。如果一个类没有重写finalize()方法,Java虚拟机会自动调用默认的finalize()方法。
-
注意,Java虚拟机并不保证在任何时刻都会卸载一个类。在某些情况下,Java虚拟机可能会选择不卸载一个类,而是将其保留在内存中以提高程序性能。例如,如果一个类被频繁使用,Java虚拟机可能会将其保留在内存中以避免重复加载。
总结
- 几个要点:
- 静态变量的定义使用final关键字,这类变量会在准备阶段直接进行初始化(除非要执行方法)。
- 直接访问父类的静态变量,不会触发子类的初始化。子类的初始化cinit调用之前,会先调用父类的cinit初始化方法。
相关文章:

JVM类的声明周期
文章目录 版权声明生命周期概述加载阶段查看内存中的对象 连接阶段连接阶段之验证连接阶段之准备连接阶段之解析 初始化阶段练习题目一练习题目二练习题目三练习题目四 使用阶段卸载阶段总结 版权声明 本博客的内容基于我个人学习黑马程序员课程的学习笔记整理而成。我特此声明…...

html将复选框变为圆形样例
html将复选框变为圆形样例 说明目录使用对勾图标实现圆形复选框原复选框html代码及默认样式取消复选框未勾选前的样式新增复选框未勾选前的样式新增复选框勾选后的样式获取复选框选中后的value值 使用CSS样式写对勾图标实现圆形复选框 说明 这里记录下用原生html实现将原复选框…...

笔记软件 Keep It mac v2.3.3中文版新增功能
Keep It mac是一款专为 Mac、iPad 和 iPhone 设计的笔记和信息管理应用程序。它允许用户在一个地方组织和管理他们的笔记、网络链接、PDF、图像和其他类型的内容。Keep It 还具有标记、搜索、突出显示、编辑和跨设备同步功能。 Keep It for mac更新日志 修复了更改注释或富文本…...

uni-app 开发的H5 定位功能部署注意事项
一、H5部署的时候,如果设计到定位功能,需要注意以下几点 1、打包部署的时候需要在Web配置-定位和地图里面勾选一个地图,并配置key 2、打包部署需要域名是https协议的,大多数现代浏览器要求在HTTPS协议下才能够访问地理位置信息&a…...

CY5-COOH脂溶性羧基荧光染料1032678-07-1
Cyanine5-COOH是一种荧光染料,它CAS号1032678-07-1,分子式为C32H39ClN2O2,分子量为519.12。Cyanine5-COOH具有良好的光稳定性和荧光亮度,可以用于生物学研究、诊断、药物筛选等领域。 Cy5-COOH (来自星戈瑞的花菁染料) 含有羧基官…...

【CSS】div 盒子居中的常用方法
<body><div class"main"><div class"box"></div></div> </body>绝对定位加 margin: auto; : <style>* {padding: 0;margin: 0;}.main {width: 400px;height: 400px;border: 2px solid #000;positio…...

Pytorch网络模型训练
现有网络模型的使用与修改 vgg16_false torchvision.models.vgg16(pretrainedFalse) # 加载一个未预训练的模型 vgg16_true torchvision.models.vgg16(pretrainedTrue) # 把数据分为了1000个类别print(vgg16_true) 以下是vgg16预训练模型的输出 VGG((features): S…...

webgoat-Path traversal
Path traversal 路径(目录)遍历是一种漏洞,攻击者能够访问或存储外部的文件和目录 应用程序运行的位置。这可能会导致从其他目录读取文件,如果是文件,则会导致读取文件 上传覆盖关键系统文件。 它是如何工作的&#…...
P8976 「DTOI-4」排列,贪心
题目背景 Update on 2023.2.1:新增一组针对 yuanjiabao 的 Hack 数据,放置于 #21。 Update on 2023.2.2:新增一组针对 CourtesyWei 和 bizhidaojiaosha 的 Hack 数据,放置于 #22。 题目描述 小 L 给你一个偶数 n 和两个整数a,b…...

使用 Python 进行自然语言处理第 5 部分:文本分类
一、说明 关于文本分类,文章已经很多,本文这里有实操代码,明确而清晰地表述这种过程,是实战工程师所可以参照和依赖的案例版本。 本文是 2023 年 1 月的 WomenWhoCode 数据科学跟踪活动提供的会议系列文章中的一篇。 之前的文章在…...

uni-app---- 点击按钮拨打电话功能点击按钮调用高德地图进行导航的功能【安卓app端】
uniapp---- 点击按钮拨打电话功能&&点击按钮调用高德地图进行导航的功能【安卓app端】 先上效果图: 1. 在封装方法的文件夹下新建一个js文件,然后把这些功能进行封装 // 点击按钮拨打电话 export function getActionSheet(phone) {uni.showAct…...

通讯录详解(静态版,动态版,文件版)
💓博客主页:江池俊的博客⏩收录专栏:C语言进阶之路👉专栏推荐:✅C语言初阶之路 ✅数据结构探索✅C语言刷题专栏💻代码仓库:江池俊的代码仓库🎉欢迎大家点赞👍评论&#x…...
在windows中搭建vue开发环境
1.环境搭建 具体环境搭建步骤参考链接 注意该博客中初始化命令: vue init webpack MyPortalProject需改为小写: vue init webpack myportalproject不然会报错 Warning: name can no longer contain capital letters2.创建第一个vueelement ui项目 …...

数字化转型:云表低代码开发助力制造业腾飞
数字化转型已成为制造业不可避免的趋势。为了应对市场快速变化、提高运营效率以及降低成本,制造业企业积极追求更加智能化、敏捷的生产方式。在这个转型过程中,低代码技术作为一种强大的工具,正逐渐崭露头角,有望加速制造业的数字…...

Linux学习之vim跳转到特定行数
参考的博客:《Vim跳到最后一行的方法》 《oeasy教您玩转vim - 14 - # 行头行尾》 《Linux:vim 中跳到首行和最后一行》 想要跳到特定行的话,可以在命令模式和正常模式进行跳转。要是对于vim的四种模式不太熟的话,可以到博客《Linu…...

详解基于Android的Appium+Python自动化脚本编写
📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…...

【马蹄集】—— 百度之星 2023
百度之星 2023 目录 BD202301 公园⭐BD202302 蛋糕划分⭐⭐⭐BD202303 第五维度⭐⭐ BD202301 公园⭐ 难度:钻石 时间限制:1秒 占用内存:64M 题目描述 今天是六一节,小度去公园玩,公园一共 N N N 个景点&am…...

大数据毕业设计选题推荐-无线网络大数据平台-Hadoop-Spark-Hive
✨作者主页:IT毕设梦工厂✨ 个人简介:曾从事计算机专业培训教学,擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…...
【jvm】虚拟机之本地方法接口与本地方法库
目录 一、本地方法1.1 说明1.2 代码示例1.3 为什么要使用native method 二、现状 一、本地方法 1.1 说明 1.一个Native Method就是一个Java调用非Java代码的接口。 2.一个Native Method是这样一个Java方法:该方法的实现由非Java语言实现,比如C。 3.这个…...

HDFS系统操作命令大全
一,前言 HDFS作为分布式存储的文件系统,有其对数据的路径表达方式 HDFS同linux系统一样,均是以/作为根目录的组织形式 linux:/usr/local/hello.txt HDFS:/usr/local/hello.txt 二,如何区分呢? L…...

测试微信模版消息推送
进入“开发接口管理”--“公众平台测试账号”,无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息: 关注测试号:扫二维码关注测试号。 发送模版消息: import requests da…...

智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...

大数据零基础学习day1之环境准备和大数据初步理解
学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 (1)设置网关 打开VMware虚拟机,点击编辑…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...
A2A JS SDK 完整教程:快速入门指南
目录 什么是 A2A JS SDK?A2A JS 安装与设置A2A JS 核心概念创建你的第一个 A2A JS 代理A2A JS 服务端开发A2A JS 客户端使用A2A JS 高级特性A2A JS 最佳实践A2A JS 故障排除 什么是 A2A JS SDK? A2A JS SDK 是一个专为 JavaScript/TypeScript 开发者设计的强大库ÿ…...
纯 Java 项目(非 SpringBoot)集成 Mybatis-Plus 和 Mybatis-Plus-Join
纯 Java 项目(非 SpringBoot)集成 Mybatis-Plus 和 Mybatis-Plus-Join 1、依赖1.1、依赖版本1.2、pom.xml 2、代码2.1、SqlSession 构造器2.2、MybatisPlus代码生成器2.3、获取 config.yml 配置2.3.1、config.yml2.3.2、项目配置类 2.4、ftl 模板2.4.1、…...
Java求职者面试指南:计算机基础与源码原理深度解析
Java求职者面试指南:计算机基础与源码原理深度解析 第一轮提问:基础概念问题 1. 请解释什么是进程和线程的区别? 面试官:进程是程序的一次执行过程,是系统进行资源分配和调度的基本单位;而线程是进程中的…...

【Post-process】【VBA】ETABS VBA FrameObj.GetNameList and write to EXCEL
ETABS API实战:导出框架元素数据到Excel 在结构工程师的日常工作中,经常需要从ETABS模型中提取框架元素信息进行后续分析。手动复制粘贴不仅耗时,还容易出错。今天我们来用简单的VBA代码实现自动化导出。 🎯 我们要实现什么? 一键点击,就能将ETABS中所有框架元素的基…...