面试官:String s = new String(“xyz“) 创建了几个对象?

文章讨论了Java中字符串对象的创建过程,特别是在JDK1.7及1.8版本的变化。在JDK1.8中,字符串常量池位于堆中。通过代码示例和字节码解析,作者指出`Strings=newString(xyz)`会创建1到2个对象,取决于字符串xyz是否已存在于常量池中。引用本身不被视为一个对象。文章还列举了一些常见的JVM指令助记符。
摘要由CSDN通过智能技术生成

这个问题相信每个学习 java 的同学都不陌生,作为一个经典的面试题,到现在工作这么多年了我真是认为挺操蛋的一个问题,在网上到现在你仍然可以看见很多讨论这个问题的人,其中不乏工作很多年的人都有争论,我认为还是有必要来说一说这个问题的。

从方法区说起

常量池存在于方法区,而方法区在jdk1.7版本前后改变比较大,所以还是先来说说方法区的演变。

在jdk1.7版本之前,常量池存在于方法区,方法区是堆的一个逻辑部分,他有一个名字叫做非堆。

339592e1372d45abb912be23ef38b2d0.jpeg

 1.7版本把字符串常量池放到了堆中。79903528ef07419abce2d73145c554de.jpeg

 而在1.8以后,则是移除了永久代,方法区概念保留,方法区的实现改为了元空间,常量池还是在堆中。

13fbdd5e26904b7fad6e5068d834db14.jpeg

 为什么要说方法区的改变,只是为了文章接下来的内容不会由于JDK的版本而产生分歧,接下来内容都会以jdk1.8版本作为基础来讨论。

String s = new String("xyz");

先来一段代码:

9bb50e41f80e4e85a797476668eed210.png

 接着我们javac编译代码,然后用javap来反编译,执行javap -c Test

520325685e7e450b8728084abf30c621.jpeg

 从结果来看,ldc命令在常量池中创建了一个"xyz"的对象,然后把他推至操作数栈顶,然后astore保存到局部变量,return返回。

接着看第二段面试题中的代码

4d45c3f964e0426eb8a034525a0e2d43.png

 同样反编译分析d2f1ac85319349b7916fee5c91cd0e48.jpeg

 

很明显,我们看到new 创建了一个String对象,同时ldc在常量池中创建了"xyz"字符串对象,之后invokespecial执行构造函数,astore_1赋值,return返回。

通过以上两个例子,可以知道String s = new String("xyz");创建了2个对象,而有些答案说的3个对象,则是把引用s也算作一个对象。

还有答案说xyz存在就创建了2个,不存在就创建了3个(包含引用s),再来测试一下。

2968dfbb1a9c4ef9825ff7f7b42d821f.png

377b3e0566a147a3b87e830ff9177195.jpeg 

 从这里,很明显的发现这就是我们例子1和2的一个结合,但是注意两次ldc后面的#2,#号代表着索引,说明第二次new String("xyz")的时候并没有重新创建xyz对象。

一些常见的指令助记符含义:

  1.nop, 什么都不做。

  2.aconst_null,将 null 推送至栈顶。

  3.iconst_i(变量数字),将 int 型 i 推送至栈顶。同理有lconst_0,fconst_0这种你应该知道什么意思了

  4.ldc,将 int,float 或 String 型常量值从常量池中推送至栈顶。

  5.iload,将指定的 int 型局部变量推送至栈顶。

  6.istore,将栈顶 int 型数值存入指定局部变量。同理astore_i代表将栈顶引用型数值存入第i个局部变量。

  7.dup,复制栈顶数值并将复制值压入栈顶。

  8.invokevirtual,调用实例方法。

  9.invokespecial,调用超类构造方法,实例初始化方法,私有方法。

  10.invokestatic,调用静态方法。

  11.invokeinterface,调用接口方法。

  12.invokedynamic,调用动态链接方法。

  13.new,创建一个对象,并将其引用值压入栈顶。

总结

到底创建了几个对象呢?

  1.如果xyz不存在,引用算对象的话,那就是3个

  2.如果xyz不存在,引用不算对象的话,那就是2个

  3.如果xyz存在,引用算对象的话,那就是2个

  4.如果xyz存在,引用不算对象的话,那就是1个

当然,我认为引用肯定是不算对象的,最终答案应该是1或者2个,这个面试题说实话不应该出现在初级面试题里。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值