点击上方 "程序员小乐"关注, 星标或置顶一起成长
后台回复“大礼包”有惊喜礼包!
关注订阅号「程序员小乐」,收看更多精彩内容
往日回顾:2020 中国软件 100 强,腾讯第二、阿里第三,他第一!
正文
这个问题相信每个学习java的同学都不陌生,作为一个经典的面试题,到现在工作这么多年了我真是认为挺操蛋的一个问题,在网上到现在你仍然可以看见很多讨论这个问题的人,其中不乏工作很多年的人都有争论,我认为还是有必要来说一说这个问题的。
从方法区说起
常量池存在于方法区,而方法区在jdk1.7版本前后改变比较大,所以还是先来说说方法区的演变。
在jdk1.7版本之前,常量池存在于方法区,方法区是堆的一个逻辑部分,他有一个名字叫做非堆。
1.7版本把字符串常量池放到了堆中。
而在1.8以后,则是移除了永久代,方法区概念保留,方法区的实现改为了元空间,常量池还是在堆中。
为什么要说方法区的改变,只是为了文章接下来的内容不会由于JDK的版本而产生分歧,接下来内容都会以jdk1.8版本作为基础来讨论。
String s = new String("xyz");
先来一段代码
public class Test {
public static void main(String[] args) {
String s = "xyz";
}
}
接着我们javac编译代码,然后用javap来反编译,执行javap -c Test
从结果来看,ldc命令在常量池中创建了一个"xyz"的对象,然后把他推至操作数栈顶,然后astore保存到局部变量,return返回。
接着看第二段面试题中的代码
public class Test {
public static void main(String[] args) {
String s = new String("xyz");
}
}
同样反编译分析
很明显,我们看到new 创建了一个String对象,同时ldc在常量池中创建了"xyz"字符串对象,之后invokespecial执行构造函数,astore_1赋值,return返回。
在公众号程序员小乐回复“Java”,获取Java面试题和答案惊喜礼包。
通过以上两个例子,可以知道String s = new String("xyz"); 创建了2个对象,而有些答案说的3个对象,则是把引用s也算作一个对象。
还有答案说xyz存在就创建了2个,不存在就创建了3个(包含引用s),再来测试一下。
public class Test {
public static void main(String[] args) {
String s = "xyz";
String s2 = new String("xyz");
}
}
一些常见的指令助记符含义:
-
nop, 什么都不做。 -
aconst_null,将 null 推送至栈顶。 -
iconst_i(变量数字),将 int 型 i 推送至栈顶。同理有lconst_0,fconst_0这种你应该知道什么意思了 -
ldc,将 int,float 或 String 型常量值从常量池中推送至栈顶。 -
iload,将指定的 int 型局部变量推送至栈顶。 -
istore,将栈顶 int 型数值存入指定局部变量。同理astore_i代表将栈顶引用型数值存入第i个局部变量。 -
dup,复制栈顶数值并将复制值压入栈顶。 -
invokevirtual,调用实例方法。 -
invokespecial,调用超类构造方法,实例初始化方法,私有方法。 -
invokestatic,调用静态方法。 -
invokeinterface,调用接口方法。 -
invokedynamic,调用动态链接方法。 -
new,创建一个对象,并将其引用值压入栈顶。
总结
到底创建了几个对象呢?
-
如果xyz不存在,引用算对象的话,那就是3个
-
如果xyz不存在,引用不算对象的话,那就是2个
-
如果xyz存在,引用算对象的话,那就是2个
-
如果xyz存在,引用不算对象的话,那就是1个
当然,我认为引用肯定是不算对象的,最终答案应该是1或者2个,这个面试题说实话不应该出现在初级面试题里。
PS:欢迎在留言区留下你的观点,一起讨论提高。如果今天的文章让你有新的启发,欢迎转发分享给更多人。
猜你还想看
BAT面试经验总结
在下面公众号后台(非本号),回复关键字「手册」获取
本文分享自微信公众号 - 程序员小乐(study_tech)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。
来源:oschina
链接:https://my.oschina.net/u/2707828/blog/4813430