JVM—类的生命周期

目录

类的生命周期

加载阶段

连接阶段

验证阶段

准备阶段

解析阶段

初始化阶段

面试题1

面试题2 


类的生命周期


类的生命周期描述了一个类加载、使用、卸载的整个过程,整体可以分为以下五个阶段。
1. 加载
2. 连接,其中又分为验证、准备、解析三个子阶段
3. 初始化
4. 使用
5. 卸载

加载阶段

一、加载阶段的第一步是类加载器通过不同的渠道,以二进制流的形式获取字节码信息,这些渠道可以通过程序员使用Java代码进行扩展,常见渠道如下。
1. 从本地磁盘上获取文件
2. 运行时通过动态代理生成,比如Spring框架
3. Applet技术通过网络获取字节码文件

二、类加载器加载类后,Java虚拟机会将字节码中的信息存储到方法区,并在方法区中生成一个 InstanceKlass 对象。这个对象保存了类的所有信息,包括实现多态等特定功能的信息。

三、Java虚拟机同时会在堆上生成与方法区中数据相似的 java.lang.Class 对象,其作用是在Java代码中获取类的信息并存储静态字段的数据(适用于JDK8及之后版本)。

堆中的对象仅包含方法区对象的部分信息。对开发者而言,只需访问堆中的 Class 对象,而无需接触方法区中的所有信息。这样,Java虚拟机可以有效控制开发者对数据的访问范围。


连接阶段

验证阶段

验证阶段的主要目的是检测Java字节码文件是否符合《Java虚拟机规范》的约束。该阶段通常无需程序员参与,主要包括以下四个部分,具体细节请参见《Java虚拟机规范》:
1. 文件格式验证:检查文件是否以0xCAFEBABE开头,以及主次版本号是否符合当前Java虚拟机版本的要求;
2. 元信息验证:确保类具有父类(即super不能为null);
3. 指令语义验证:检查程序执行指令的正确性,例如方法内指令是否跳转到合法的位置;
4. 符号引用验证:确认是否尝试访问其他类中的private方法等。

准备阶段

准备阶段负责为静态变量(static)分配内存并设置初始值。在这一阶段,每种基本数据类型和引用数据类型都有其特定的默认值。而在初始化阶段,静态变量的值才会被更新为代码中指定的值。需要注意的是,对于被 final 修饰的基本数据类型静态变量,准备阶段会直接将其赋予代码中指定的值。这一阶段的重要性在于,它确保所有静态变量在类加载时都有一个一致的状态,为后续的初始化提供基础。

解析阶段

解析阶段的主要任务是将常量池中的符号引用替换为直接引用。在字节码文件中,符号引用通过编号访问常量池的内容,而直接引用则使用内存地址进行访问。这一转换提高了访问效率,减少了查找开销,使得Java虚拟机能够更快速地定位和使用数据。


初始化阶段

初始化阶段会执行字节码文件中clinit(class init 类的初始化)方法的字节码指令,包含了静态代码块中的代码,并为静态变量赋值。

public class Demo1 {

    public static int value = 1;
    static {
        value = 2;
    }
   
    public static void main(String[] args) {

    }
}

以上代码编译成字节码文件之后,会生成以下三个方法:
1、init方法:会在对象初始化时执行
2、main方法:主方法
3、clinit方法:类的初始化阶段执行

接下来分析clinit方法中的字节码指令

1、iconst_1,将常量1放入操作数栈。此时栈中只有1这个数;
2、putstatic指令会将操作数栈上的数弹出来,并放入堆中静态变量的位置,字节码指令中#2指向了常量池中的静态变量value,在解析阶段会被替换成变量的地址;
3、前面两步操作相当于执行value=1;后面两步操作同理,执行value=2,将堆上的value赋值为2。

public class Demo1 {

    static {
        value = 2;
    }
    public static int value = 1;
   

    public static void main(String[] args) {

    }
}

/*  原代码顺序
    public static int value = 1;
    static {
        value = 2;
    }
*/

如果将代码的位置互换,字节码指令的位置也会发生变化,如下图。 这样初始化结束之后,最终value的值就变成了1而不是2。


引起clinit执行的几种情况
1. 访问类的静态变量或静态方法;
注意:如果变量被 final 修饰且等号右侧的值是常量,则不会触发初始化。
2. 调用 Class.forName(String className) 时;
3. 创建该类的对象(使用 new 关键字);
4. 执行该类的 main 方法时。
Tip:添加-XX:+TraceClassLoading 参数可以打印出加载并初始化的类

clinit不会执行的几种情况
1.无静态代码块且无静态变量赋值语句;
2.有静态变量的声明,但是没有赋值语句;(如:public static int a;)
3.静态变量的定义使用final关键字,这类变量会在准备阶段直接进行初始化;
(如:public final static int a= 10;)
4.数组的创建不会导致数组中元素的类进行初始化。


面试题1

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");
    }
}

以上代码的输出结果是什么?
1、执行main方法之前,会先执行该类的clinit方法。clinit方法中执行了静态代码块中的方法,因此输出D;
2、之后走main方法,输出A;
3、随后创建两个对象,原则上会执行两次对象初始化的指令,但该对象在第一步时已经被加载并初始化,所以此处不再加载与初始化,而是走构造方法;
4、构造代码块优先于构造方法执行,因此先输出C,再输出B;创建两个对象,因此输出CBCB;
5、所以代码输出的结果为DACBCB


面试题2

public class Demo01 {
    public static void main(String[] args) {
        new B02();
        System.out.println(B02.a);
    }
}

class A02{
    static int a = 0;
    static {
        a = 1;
    }
}

class B02 extends A02{
    static {
        a = 2;
    }
}

以上代码的输出结果是什么?
1、调用new创建对象,需要初始化B02,有父类的情况优先初始化父类;
2、执行A02的初始化代码,最终将a赋值为1;
3、父类初始完成,进行本类B02初始化,将a赋值为2;
4、最终输出2。

变形
如果将new B02();注释掉会怎么样?
分析步骤:
1、访问父类的静态变量,只初始化父类,不初始本类;
2、执行A02的初始化代码,将a赋值为1;
3、最终输出1。

总结
1、直接访问父类的静态变量,不会触发子类的初始化。
2、子类的初始化clinit调用之前,会先调用父类的clinit初始化方法

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python学习的自我理解和想法(21)

学的是b站的课程(千锋教育),跟老师写程序,不是自创的代码! 今天是学Python的第21天,学的内容是文件的操作。开学了,时间不多,写得不多,见谅。 目录 1.文件 (1).参数…

Tcp_Sever(线程池版本的 TCP 服务器)

Tcp_Sever(线程池版本的 TCP 服务器) 前言1. 功能介绍及展示1.1 服务端连接1.2 客户端连接(可多个用户同时在线连接服务端)1.3 功能服务1.3.1 defaultService(默认服务)1.3.2 transform(大小写转…

Rust与Javascript的使用对比

一、常量 RustJavascriptletconst 二、变量 RustJavascriptlet mutlet / var 三、常用打印 RustJavascriptprintln!(“换行”);console.log(‘hello’);print!(“不换行”);console.info(‘信息’);-console.error(‘错误’);-console.warn(‘警告’); 四、定义字符串 R…

开放式耳机哪个品牌音质好?高评分爆款开放式耳机推荐!

一直活跃在蓝牙耳机圈子里的我,对各种类型的耳机多少都有自己的看法,完全可以说是个耳机狂热者。近几年,开放式蓝牙耳机愈发火爆。开放式耳机不是任何品牌都能轻松做好的产品,特别是音质,它涵盖了核心单元技术等诸多方…

负载均衡服务器攻击怎么解决最有效?

负载均衡服务器攻击怎么解决最有效?常见的有效解决方法包括:使用SYNCookie机制、限制ICMP包速率、基于源IP的连接速率限制、检测并丢弃异常IP包、配置访问控制列表(ACL)、设置虚拟服务器/服务器连接数量限制、设置HTTP并发请求限制…

ABAQUS应用11——支座弹簧

文章目录 0、背景1、ABAQUS中几类弹簧的简介2、SPRING1的性质初探 0、背景 1、ABAQUS中几类弹簧的简介 先说参考来源,ABAQUS2016的帮助文档里第4卷,32.1.1节,有三种弹簧(SPRING1 、SPRING2 以及SPRINGA)。 三种弹簧里…

C for Graphic:视差渲染(一)

记录一下最近优化场景的做法:视差渲染 原理:通过视口坐标的变化,观察不同采样画面的功能,画面的载体为低模平面 我早期工作,在小作坊全栈的时候,做过一段时间web开发,做了一个古董藏…

【传知代码】机器学习在情绪预测中的应用(论文复现)

在科技迅猛发展的今天,我们不仅在追求更强大的计算能力和更高的精度,还希望我们的机器能够理解和回应我们复杂的情感世界。设想一下,当你面对挫折时,设备不仅能识别你的情绪,还能以一种富有同情心和洞察力的方式作出反…

开放式耳机哪个牌子好?开放式蓝牙耳机排行榜分享

​耳机已经成为我们日常生活中的必需品,但长时间佩戴传统入耳式耳机可能会导致耳朵不适,甚至影响健康。为了应对这一挑战,开放式耳机应运而生。这类耳机不侵入耳道,有效减轻了耳朵的压力,同时减少了感染风险&#xff0…

fmql之Linux中I2C总线框架

正点原子第44章 I2C zynq I2C pcf8563芯片 我们用的是ds3231. Linux I2C总线框架 I2C总线驱动 这部分内容是半导体厂商编写的。 I2C总线设备 zynq I2C适配器驱动 I2C设备驱动编写 使用设备树 代码编写 设备树修改 设备驱动编写 因为用的是ds3231,所以先找…

使用 PyTorch 构建 LSTM 股票价格预测模型

目录 引言准备工作1. 训练模型(train.py)2. 模型定义(model.py)3. 测试模型和可视化(test.py)使用说明模型调整结论 引言 在金融领域,股票价格预测是一个重要且具有挑战性的任务。随着深度学习…

1024软件推荐-rubick

开源的插件化桌面端效率工具箱。插件是基于 npm 进行安装和卸载,非常轻便。插件数据支持 webdav 多端同步,非常安全。支持内网部署,可二次定制化开发,非常灵活。 前言 rubick 之前的插件管理,依托于云服务器存储&…

滴水逆向三期笔记与作业——02C语言——13 指针(3)(4)

滴水逆向三期笔记与作业——02C语言——13 指针3、4 一、模拟实现CE的数据搜索功能 OneNote迁移 一、模拟实现CE的数据搜索功能 //其中有0xAA,超过有符号char范围,在vscode中会报错,所以使用unsigned char unsigned char data[100] {0x00,0…

一起搭WPF架构之完结总结篇

一起搭WPF架构之完结总结篇 前言设计总结设计介绍页面一页面二页面三 结束 前言 整体基于WPF架构,根据自己的需求简单设计与实现了衣橱的数据统计、增加与读取数据、并展示数据的小软件。我知道自己在设计方面还有很多不足,暂时先做到这里了&#xff0c…

gbase8s权限管理

一 权限分类 分片级权限(分片表) 表引用 类型级权限 例程级权限 语言级权限 序列级权限 等... 其中常用的为 数据库级权限,表级权限,序列级权限以及例程级权限 二 权限控制 当创建一个用户时,该用户没有任何权…

为了数清还有几天到周末,我用python绘制了日历

日历的秘密 昨天,在看小侄子写作业的时候,发现了一个秘密:他在“演算纸”(计算数学题用的草纸)上画了非常多的日历。对此我感到了非常的困惑,“这是做什么的?” 后来,经过了我不懈…

机器学习面试笔试知识点-线性回归、逻辑回归(Logistics Regression)和支持向量机(SVM)

机器学习面试笔试知识点-线性回归、逻辑回归Logistics Regression和支持向量机SVM 一、线性回归1.线性回归的假设函数2.线性回归的损失函数(Loss Function)两者区别3.简述岭回归与Lasso回归以及使用场景4.什么场景下用L1、L2正则化5.什么是ElasticNet回归6.ElasticNet回归的使…

【设计模式】MyBatis 与经典设计模式:从ORM到设计的智慧

作者:后端小肥肠 🍇 我写过的文章中的相关代码放到了gitee,地址:xfc-fdw-cloud: 公共解决方案 🍊 有疑问可私信或评论区联系我。 🥑 创作不易未经允许严禁转载。 姊妹篇: 【设计模式】揭秘Spri…

计算机网络:数据链路层 —— 以太网(Ethernet)

文章目录 局域网局域网的主要特征 以太网以太网的发展100BASE-T 以太网物理层标准 吉比特以太网载波延伸物理层标准 10吉比特以太网汇聚层交换机物理层标准 40/100吉比特以太网传输媒体 局域网 局域网(Local Area Network, LAN)是一种计算机网络&#x…

GitLab-删除仓库分支(删除远程分支)

进入对应仓库选择对应的分支进行删除操作。