工作10年后,再看String s = new String("xyz") 创建了几个对象?

梦想的初衷 提交于 2021-02-19 02:28:03

点击上方 "程序员小乐"关注, 星标或置顶一起成长

后台回复“大礼包”有惊喜礼包!

关注订阅号「程序员小乐」,收看更多精彩内容

每日英文

When a person cannot answer directly to ur question, probably the answer is too painful for you to know or too hard for them to admit.

如果一个人不正面回答你的问题,可能是说出来你难以接受,或者他们不愿意承认。

每日掏心话

一生中,最光辉的一天并非功成名就的那一天,而是从悲叹与绝望中产生对人生挑战与勇敢迈向意志的那一天。

:艾小仙 | 责编:乐乐


后端架构师(ID:study_tech) 第 1085 次推文


往日回顾: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");
    }
}


从这里,很明显的发现这就是我们例子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个,这个面试题说实话不应该出现在初级面试题里。


PS:欢迎在留言区留下你的观点,一起讨论提高。如果今天的文章让你有新的启发,欢迎转发分享给更多人。


欢迎加入后端架构师交流群,在后台回复“学习”即可。

猜你还想看

阿里、腾讯、百度、华为、京东最新面试题汇集

员工踩点上下班被HR约谈,网友:按时上下班,天经地义

再见!Postman

Java实现单链表、栈、队列三种数据结构


BAT面试经验总结


在下面公众号后台(非本号),回复关键字「手册」获取



嘿,你在看吗

本文分享自微信公众号 - 程序员小乐(study_tech)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!