从JVM角度看继承

news2025/1/21 5:58:16

从JVM角度看继承

最近重读了周志明老师的《深入理解JAVA虚拟机》一书,看完大有收获,但仍对继承情况下对象内存布局有所疑惑,所以查阅资料,结合本书进行分析

参考文档:

【深入理解JVM】:Java类继承关系中的初始化顺序

从JVM内存机制理解 java 的继承

继承的对象内存布局

对象在堆内存中的存储布局可以划分为三个部分:

对象头(Header)

实例 数据(Instance Data)

对齐填充(Padding)

书中原文:

实例数据部分是对象真正存储的有效信息,即我们在程序代码里面所定义的各种类型的字段内容,无论是从父类继承下来的,还是在子类中定义的字段都必须记录起来。这部分的存储顺序会 受到虚拟机分配策略参数(-XX:FieldsAllocationStyle参数)和字段在Java源码中定义顺序的影响。 HotSpot虚拟机默认的分配顺序为longs/doubles、ints、shorts/chars、bytes/booleans、oops(Ordinary Object Pointers,OOPs),从以上默认的分配策略中可以看到,相同宽度的字段总是被分配到一起存 放,在满足这个前提条件的情况下,在父类中定义的变量会出现在子类之前。如果HotSpot虚拟机的 +XX:CompactFields参数值为true(默认就为true),那子类之中较窄的变量也允许插入父类变量的空 隙之中,以节省出一点点空间。

可以看到继承下来的非静态成员变量是存在于子类的,而静态成员变量存在于方法区

初始化的时机

原文:

关于在什么情况下需要开始类加载过程的第一个阶段“加载”,《Java虚拟机规范》中并没有进行
强制约束,这点可以交给虚拟机的具体实现来自由把握。但是对于初始化阶段,《Java虚拟机规范》
则是严格规定了有且只有六种情况必须立即对类进行“初始化”(而加载、验证、准备自然需要在此之
前开始):
1)遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,如果类型没有进行过初始化,则需要先触发其初始化阶段。能够生成这四条指令的典型Java代码场景有:
· 使用new关键字实例化对象的时候。
· 读取或设置一个类型的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候。
· 调用一个类型的静态方法的时候。
2)使用java.lang.reflect包的方法对类型进行反射调用的时候,如果类型没有进行过初始化,则需要先触发其初始化。
3)当初始化类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
4)当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
5)当使用JDK 7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为REF_getStatic、REF_putStatic、REF_invokeStatic、REF_newInvokeSpecial四种类型的方法句柄,并且这个方法句柄对应的类没有进行过初始化,则需要先触发其初始化。
6)当一个接口中定义了JDK 8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。

对于这六种会触发类型进行初始化的场景,《Java虚拟机规范》中使用了一个非常强烈的限定语——“有且只有”,这六种场景中的行为称为对一个类型进行主动引用。除此之外,所有引用类型的方式都不会触发初始化,称为被动引用。

执行顺序

  1. 父类静态代码区和父类静态成员
  2. 子类静态代码区和子类静态成员
  3. 父类非静态代码区和普通成员
  4. 父类构造函数
  5. 子类非静态代码区和普通成员
  6. 子类构造函数

原文:

()方法与类的构造函数(即在虚拟机视角中的实例构造器()方法) 不同,它不需要显式地调用父类构造器,Java虚拟机会保证在子类的()方法执行前,父类的0方法已经执行完毕。因此在Java虚拟机中第一个被执行的()方法的类型肯定是java.lang.Object。
由于父类的()方法先执行,也就意味着父类中定义的静态语句块要优先于子类的变量赋值操作

ok 1,2步的顺序没问题

原文:

例如前文多次登场的实例构造器()方法和类构造器()方法就是在这个阶段被添加到语
法树之中的。请注意这里的实例构造器并不等同于默认构造函数,如果用户代码中没有提供任何构造
函数,那编译器将会添加一个没有参数的、可访问性(public、protected、private或)与当前
类型一致的默认构造函数,这个工作在填充符号表阶段中就已经完成。()和()这两个构造
器的产生实际上是一种代码收敛的过程,

编译器会把

语句块(对于实例构造器而言是“{}”块,对于类构造器而言是“static{}”块)、

变量初始化(实例变量和类变量)、

调用父类的实例构造器(仅仅是实例构造器,()方法中无须调用父类的()方法,Java虚拟机会自动保证父类构造器的正确执行,但在()方法中经常会生成调用java.lang.Object的()方法的代码)

等操作收敛到()和()方法之中,并且保证无论源码中出现的顺序如何,都一定是按先执行父类的实例构造器,然后初始化变量,最后执行语句块的顺序进行,上面所述的动作由Gen::normalizeDefs()方法来实现。

除了生成构造器以外,还有其他的一些代码替换工作用于优化程序某些逻辑的实现方式,如把字符串的加操作替换为StringBuffer或StringBuilder(取决于目标代码的版本是否大于或等于JDK 5)的append()操作,等等。

由上文可以看到,和的包括范围,有些人认为阶段仅仅是执行构造方法,这明显是错误的,原文中已经表明()和()方法是一些操作的集合

方法:()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的

方法:除了()方法包括的操作,剩下的包括实例变量初始化,非静态代码块,构造方法(先执行父类,再执行子类)

4,5步的顺序也ok

先执行父类的实例构造器,然后初始化变量,最后执行语句块的顺序

1,2,3,5步骤中代码块和成员变量的初始化也有先后顺序,一般是先初始化变量,后静态代码块

至于和的顺序,则是因为是执行在类加载的的初始化流程中,可以说初始化的过程就是,而的执行必须在类加载之后,所以这两者的顺序也是确定的

默认构造器

image-20230908104039730

自定义构造器

image-20230907181016769

可以看到,无论是默认构造器,还是用户定义构造器,均先调用父类的构造函数Object

L0
    LINENUMBER 6 L0
    ALOAD 0
    INVOKESPECIAL java/lang/Object.<init> ()V

调用构造函数是否一定会产生实例?

我认为不是

比如子类调用抽象父类的构造函数,父接口的构造函数,会产生子类实例,但会产生父级的实例吗?

这显然是不会的

抽象类的构造方法为自定义构造器时,子类需要显式调用

image-20230908160437762

image-20230908160327720

关于java构造函数 的错误 there is no default constructor available in

B类继承A类,在B类的构造器中,会隐式存在 super(),用来调用父类无参构造器
而父类A中没有无参构造器,因为A中已经定义了有参构造器(在A中,如果没有定义有参构造器,就会有默认的无参构造器;但如果定义了有参构造器,就没有默认的无参构造器)。所以会显示上述错误。

解决方法:

**法一:**在A类(即父类)中,添加一个无参构造器
**法二:**在B类(子类)的有参构造器中,添加一个super(m);即可
B(int m){ super(m); i = 2; }

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

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

相关文章

算法训练营day45|动态规划 part07:完全背包 (LeetCode 70. 爬楼梯(进阶)、322. 零钱兑换、279.完全平方数)

文章目录 70. 爬楼梯(进阶)(求排列方法数)思路分析代码实现 322. 零钱兑换(求等于背包重量的最小物品数)思路分析代码实现思考总结 279.完全平方数 (求等于背包重量的最小物品数)思路分析代码实现 70. 爬楼梯(进阶)(求排列方法数) 题目链接&#x1f525; 假设你正在爬楼梯。需…

jquery jstree的懒加载

效果如下 使用jquery的jstree组件 1.前端准备工作 1.1引入jstree样式和js <link rel"stylesheet" href"/public/vendor/jstree/jstree.css"> <div id"departmentJstree"></div> <script src"/public/vendor/jstree…

Dominosa/数邻(2) | C++ | BFS

目录 一、Dominosa简介二、题目描述三、编程思路四、完整代码 一、Dominosa简介 Dominosa&#xff0c;中文名称为数邻&#xff0c;是一种棋盘游戏&#xff0c;基于骨牌的排列和匹配来进行。它是从骨牌游戏中发展而来的&#xff0c;在骨牌的基础上添加了一些规则和难度。具体的游…

canape中快速配置需要录制参数的技巧

以前在车里录制数据时&#xff0c;大量融合数据一个一个拖拽 &#xff0c;不仅慢&#xff0c;有时心不细的话&#xff0c;还会漏选、挑错。 用正则表达式的挑选方法&#xff0c;可以既快速又准确的挑出所需数据。 以下蓝色字体是操作方法&#xff1a; 正则表达式 示例&#xff…

QTableView通过setColumnWidth设置了列宽无效的问题

在用到QT的QTableView时&#xff0c;为了显示效果&#xff0c;向手动的设置每一列的宽度&#xff0c;但是如下的代码是无效的。 ui->tableView->setColumnWidth(0,150);ui->tableView->setColumnWidth(1,150);ui->tableView->setColumnWidth(2,150);ui->t…

docker-compose安装nginx

基于docker-compose安装nginx 目录 一、目录结构 1、docker-compose.yml 2、nginx.conf 3、default.conf 4、index.html 二、访问测试 一、目录结构 1、docker-compose.yml version: 3 services:nginx:image: registry.cn-hangzhou.aliyuncs.com/zhengqing/nginx:1.21.1…

Linux指令二【进程,权限】

进程是一个具有一定独立功能的程序在一个数据集上的一次动态执行的过程&#xff0c;是操作系统进行 资源分配和调度的一个独立单位&#xff0c;是应用程序运行的载体。 一、进程基本指令 1.ps&#xff1a;当前的用户进程 ps 只显示隶属于自己的进程状态ps -aux 显示所有进程…

C/C++计算(a+b)c的值 2019年9月电子学会青少年软件编程(C/C++)等级考试一级真题答案解析

目录 C/C计算(ab)c的值 一、题目要求 1、编程实现 2、输入输出 二、解题思路 1、案例分析 三、程序代码 四、程序说明 五、运行结果 六、考点分析 C/C计算(ab)c的值 2019年9月 C/C编程等级考试一级编程题 一、题目要求 1、编程实现 给定3个整数a、b、c&#xff0…

错误: 找不到或无法加载主类 Main

在用git回退到上个版本后发现&#xff0c;无法运行项目并提示 错误: 找不到或无法加载主类 Main 可以看到Main前面的图标也是号。 查了半天没有解决&#xff0c;问了个大佬&#xff0c;大佬一下就解决掉了&#xff0c;本文记录下解决过程。 错误原因是编辑器无法找到代码位置&…

java(五)继承和多态,抽象类与接口,异常(javaSE完)

八、继承和多态&#xff08;重要&#xff09; 对于java来说&#xff0c;最重要的就是面对对象&#xff0c;而如何体现这个&#xff0c;在其中三个概念极为重要&#xff0c;封装、继承、多态而无论考试还是面试通常都会考察这几个概念及其原理用法。 1.继承 面向对象思想中提…

Java“牵手”淘宝商品详情数据,淘宝商品详情API接口,淘宝API接口申请指南

淘宝平台商品详情接口是开放平台提供的一种API接口&#xff0c;通过调用API接口&#xff0c;开发者可以获取淘宝商品的标题、价格、库存、月销量、总销量、库存、详情描述、图片等详细信息 。 获取商品详情接口API是一种用于获取电商平台上商品详情数据的接口&#xff0c;通过…

Windows 安装 MariaDB 数据库

之前一直使用 MySQL&#xff0c;使用 MySQL8.0 时候&#xff0c;占用内存比较大&#xff0c;储存空间好像也稍微有点大&#xff0c;看到 MariaDB 是用来代替 MySQL 的方案&#xff0c;之前用着也挺得劲&#xff0c;MySQL8.0 以上好像不能去导入低版本的 sql&#xff0c;或者需要…

MySQL下载安装环境变量配置,常用命令

一、下载安装 mysql官网 下载连接 这个是下载图形安装 https://dev.mysql.com/downloads/installer/ 这个是下载免图形安装 https://dev.mysql.com/downloads/mysql/ 担心个别宝宝没有账号&#xff0c;这边也提供一下&#xff0c;方便下载&#xff1a; 账户&#xff1a;1602404…

宝兰德部署包特别慢部署超时失败问题解决

部署慢问题解决&#xff1a; 默认这块是空的&#xff0c;我改成 G1好像就快了。 刚开始是部署超时也是基于第一个问题的原因&#xff0c;如果你还有其他原因&#xff0c;但是查看服务日志一直显示超时&#xff0c;那可能是你看错日志文件了。 宝兰德操作日志 /data/besapp…

Matlab图像处理-迭代式阈值选择法

基本思想 迭代式阈值选择法的基本思想是&#xff1a;开始时&#xff0c;选择一个阈值作为初始估计值&#xff0c;然后按某种策略不断地改进这一估计值&#xff0c;直到满足给定的准则为止。在迭代过程中&#xff0c;关键之处在于选择什么样的阈值改进策略。好的阈值改进策略应…

学生免费申请IDEA使用流程

IntelliJ IDEA一般简称IDEA&#xff0c;是Java编程语言开发的集成环境&#xff0c;在业界被公认为最好的Java开发工具。 1 IDEA官网下载 1.1 官网地址 https://www.jetbrains.com/idea/ 1.2 IDEA下载 访问官网&#xff0c;单击download按钮&#xff0c;下载“IntelliJ IDE…

【疑难杂症】使用xshell连接云服务器连接不上

目录 【1】使用xshell连接云服务器连接不上 【1.1】解决方法一 【1.2】解决方法二 【1】使用xshell连接云服务器连接不上 Centos7使用xshell连接提示"ssh服务器拒绝了密码 请再试一次"。 问题如图所示&#xff0c;新安装了一台Centos7服务器&#xff0c;使用ssh连…

200行代码实现canvas九宫格密码锁

现在很多app&#xff0c;在一些隐私页面&#xff0c;往往都会加入二次验证&#xff0c;例如银行app、支付宝理财和我的页面&#xff0c;一般会有「九宫格密码」和指纹密码。 今天我们用canvas来写一个九宫格手势密码锁&#xff0c;大概就是下面这样。 思路 准备一个正方形画布…

四、子向父传值,展示项目经验

简介 发布订阅消息,子向父传值,展示项目经验详细信息。欢迎访问个人的简历网站预览效果 本章涉及修改与新增的文件:Fifth.vue、App.vue、utils、Project.ts 一、创建项目数据 在 src 目录下新建一个 utils 文件夹 ,再创建一个 Project.ts 文件 // 项目经验的详细数据 c…

day56补

583. 两个字符串的删除操作 力扣题目链接(opens new window) 给定两个单词 word1 和 word2&#xff0c;找到使得 word1 和 word2 相同所需的最小步数&#xff0c;每步可以删除任意一个字符串中的一个字符。 示例&#xff1a; 输入: "sea", "eat"输出: …