贡献一道超高套路JVM面试题

大家好,我是江湖人送外号[道格牙]的子牙老师。

最近在手撸JVM实现OOP的封装机制的时候,有个问题引起了我的眉头紧锁。代码如下
public class Test_2 {public static void main(String[] args) {Test_2 obj = new Test_2();}}

于是我把这个让我眉头紧锁的问题提炼成了一个面试题:这段创建对象的代码,在JVM内部创建了几个对象?

正方

两个:一个是Test_2对象,一个是Object对象。

会创建Test_2对象就不用说了。为什么会创建Object对象呢?因为Java中所有的类都继承自Object类,所有对象的创建都会调用类的构造方法,而这些构造方法中都会调用Object的构造方法。如图
在这里插入图片描述
图说明:这个图是Test_2的构造方法对应的字节码指令。有小伙伴不了解字节码的建议补一补,这玩意很重要。JVM的黑科技玩的就是这个!

逻辑紧凑,毫无破绽,有木有?多少小伙伴也是这样认为的留言区举个爪。

如何证明创建了两个对象?不知道!

反方

一个:只会创建Test_2对象。

那调用Object的构造函数干吗?反方内心开始慌乱~
如果Object类有属性,不创建Object对象,如何访问Object类中的属性?反方内心动摇:我支持正方观点。
在这里插入图片描述
你跑来反方干什么?反奸!

正解

为什么我把创建两个对象定义为正方呢?呵,惭愧,我之前也是这样认为的。所以,答案是反方观点是正确的。那那两个问题怎么解释?呵,稳住,听我娓娓道来。

调用父类的构造函数,目的是什么?解释这个问题前,先说构造函数是干什么的:完成非静态属性的赋值。所以,调用父类的构造函数,目的是为了完成父类中的非静态属性的赋值。这是第一个问题。

接下来第二个问题:不创建父类对象如何访问父类的属性?其实这个问题的本质是:父类中的属性到底存储在哪?其实父类的属性是存储在子类对象上的,所以没必要创建父类对象!是不是有小伙伴不服气啦,没事,上图
在这里插入图片描述
你也可以通过HSDB查看JVM内存,结论是一样的
在这里插入图片描述
再多讲一个问题吧:父类的属性是如何存储到子类对象上的?因为调用父类对象是以子类对象身份调用的,所以父类构造函数中的非静态属性赋值语句,作用的都是子类对象。
在这里插入图片描述
如何证明?跟着我学习过JVM的小伙伴应该知道吧。借助Idea的调试功能。如图
在这里插入图片描述
在这里插入图片描述
为什么会出现两种不同的观点呢?我后面想了想:大部分小伙伴把调用构造函数等同于创建对象。其实从函数的本质来看:函数只是接受外面传参,执行相关逻辑,传出参数。构造函数本质上还是函数,并不具备创建对象的能力。

我有话说

答应我,如果你学会了,不要拿这道题去为难别人好吗?如果你实在忍不住,不要告诉别人是子牙老师那边学来的。我不想承担无谓的仇恨:NND,这是哪个抽风的面试官想到的面试题!同是程序猿,相煎何太急!

看到这篇文章的小伙伴,如果你在哪里面试遇到了这道面试题,请留言告诉我。我再发几个声讨!我与罪恶不同戴天!

推荐阅读

1、啊?i++与++i居然有这么多学问在里面
2、内存编织技术,JVM对内存的又一次压榨
3、困扰了你大半辈子的STW,今天总算可以毕业了

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值