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

沙海 2021年3月30日04:04:22杂谈 Java评论32字数 2136阅读7分7秒阅读模式
摘要

速读摘要

速读摘要文章源自JAVA秀-https://www.javaxiu.com/7462.html

这个问题相信每个学习java的同学都不陌生,作为一个经典的面试题,到现在工作这么多年了我真是认为挺操蛋的一个问题,在网上到现在你仍然可以看见很多讨论这个问题的人,其中不乏工作很多年的人都有争论,我认为还是有必要来说一说这个问题的。在jdk1.7版本之前,常量池存在于方法区,方法区是堆的一个逻辑部分,他有一个名字叫做非堆。创建了2个对象,而有些答案说的3个对象,则是把引用s也算作一个对象。同理astore_i代表将栈顶引用型数值存入第i个局部变量。文章源自JAVA秀-https://www.javaxiu.com/7462.html

原文约 1532 | 图片 8 | 建议阅读 4 分钟 | 评价反馈文章源自JAVA秀-https://www.javaxiu.com/7462.html

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

点击关注 ? 程序IT圈 文章源自JAVA秀-https://www.javaxiu.com/7462.html

点击上方蓝色“程序IT圈”,选择“设为星标”文章源自JAVA秀-https://www.javaxiu.com/7462.html

回复“666”获取独家整理的学习资料!文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

转自:艾小仙文章源自JAVA秀-https://www.javaxiu.com/7462.html

这个问题相信每个学习java的同学都不陌生,作为一个经典的面试题,到现在工作这么多年了我真是认为挺操蛋的一个问题,在网上到现在你仍然可以看见很多讨论这个问题的人,其中不乏工作很多年的人都有争论,我认为还是有必要来说一说这个问题的。文章源自JAVA秀-https://www.javaxiu.com/7462.html

从方法区说起

文章源自JAVA秀-https://www.javaxiu.com/7462.html

常量池存在于方法区,而方法区在jdk1.7版本前后改变比较大,所以还是先来说说方法区的演变。文章源自JAVA秀-https://www.javaxiu.com/7462.html

在jdk1.7版本之前,常量池存在于方法区,方法区是堆的一个逻辑部分,他有一个名字叫做非堆文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

1.7版本把字符串常量池放到了堆中。文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

而在1.8以后,则是移除了永久代,方法区概念保留,方法区的实现改为了元空间,常量池还是在堆中。文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

为什么要说方法区的改变,只是为了文章接下来的内容不会由于JDK的版本而产生分歧,接下来内容都会以jdk1.8版本作为基础来讨论。文章源自JAVA秀-https://www.javaxiu.com/7462.html

String s = new String("xyz");

文章源自JAVA秀-https://www.javaxiu.com/7462.html

先来一段代码文章源自JAVA秀-https://www.javaxiu.com/7462.html

public class Test {    public static void main(String[] args) {        String s = "xyz";    }}

接着我们javac编译代码,然后用javap来反编译,执行javap -c Test文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

从结果来看,ldc命令在常量池中创建了一个"xyz"的对象,然后把他推至操作数栈顶,然后astore保存到局部变量,return返回。文章源自JAVA秀-https://www.javaxiu.com/7462.html

接着看第二段面试题中的代码文章源自JAVA秀-https://www.javaxiu.com/7462.html

public class Test {    public static void main(String[] args) {        String s = new String("xyz");    }}

同样反编译分析文章源自JAVA秀-https://www.javaxiu.com/7462.html

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

很明显,我们看到new 创建了一个String对象,同时ldc在常量池中创建了"xyz"字符串对象,之后invokespecial执行构造函数,astore_1赋值,return返回。文章源自JAVA秀-https://www.javaxiu.com/7462.html

通过以上两个例子,可以知道String s = new String("xyz"); 创建了2个对象,而有些答案说的3个对象,则是把引用s也算作一个对象。文章源自JAVA秀-https://www.javaxiu.com/7462.html

还有答案说xyz存在就创建了2个,不存在就创建了3个(包含引用s),再来测试一下。文章源自JAVA秀-https://www.javaxiu.com/7462.html

public class Test {    public static void main(String[] args) {        String s = "xyz";        String s2 = new String("xyz");    }}

工作10年后,再看String s = new String(“xyz”) 创建了几个对象?文章源自JAVA秀-https://www.javaxiu.com/7462.html

从这里,很明显的发现这就是我们例子1和2的一个结合,但是注意两次ldc后面的#2,#号代表着索引,说明第二次new String("xyz")的时候并没有重新创建xyz对象。文章源自JAVA秀-https://www.javaxiu.com/7462.html

一些常见的指令助记符含义:文章源自JAVA秀-https://www.javaxiu.com/7462.html

  1. nop, 什么都不做。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  2. aconst_null,将 null 推送至栈顶。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  3. iconst_i(变量数字),将 int 型 i 推送至栈顶。同理有lconst_0,fconst_0这种你应该知道什么意思了文章源自JAVA秀-https://www.javaxiu.com/7462.html

  4. ldc,将 int,float 或 String 型常量值从常量池中推送至栈顶。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  5. iload,将指定的 int 型局部变量推送至栈顶。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  6. istore,将栈顶 int 型数值存入指定局部变量。同理astore_i代表将栈顶引用型数值存入第i个局部变量。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  7. dup,复制栈顶数值并将复制值压入栈顶。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  8. invokevirtual,调用实例方法。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  9. invokespecial,调用超类构造方法,实例初始化方法,私有方法。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  10. invokestatic,调用静态方法。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  11. invokeinterface,调用接口方法。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  12. invokedynamic,调用动态链接方法。文章源自JAVA秀-https://www.javaxiu.com/7462.html

  13. new,创建一个对象,并将其引用值压入栈顶。文章源自JAVA秀-https://www.javaxiu.com/7462.html

总结

文章源自JAVA秀-https://www.javaxiu.com/7462.html

到底创建了几个对象呢?文章源自JAVA秀-https://www.javaxiu.com/7462.html

  1. 如果xyz不存在,引用算对象的话,那就是3个文章源自JAVA秀-https://www.javaxiu.com/7462.html

  2. 如果xyz不存在,引用不算对象的话,那就是2个文章源自JAVA秀-https://www.javaxiu.com/7462.html

  3. 如果xyz存在,引用算对象的话,那就是2个文章源自JAVA秀-https://www.javaxiu.com/7462.html

  4. 如果xyz存在,引用不算对象的话,那就是1个文章源自JAVA秀-https://www.javaxiu.com/7462.html

当然,我认为引用肯定是不算对象的,最终答案应该是1或者2个,这个面试题说实话不应该出现在初级面试题里。文章源自JAVA秀-https://www.javaxiu.com/7462.html

热门内容:记住看小电影前一定要检查网址是不是 HTTPS 的,不然…996高薪,955底薪,你选哪个?互联网大厂的薪酬一览淘宝二面,面试官居然把TCP三次握手问的这么详细MyBatis-Plus常用API全套教程,看完没有不懂的
文章源自JAVA秀-https://www.javaxiu.com/7462.html

明天见(。・ω・。)ノ♡文章源自JAVA秀-https://www.javaxiu.com/7462.html

继续阅读
速蛙云 - 极致体验,强烈推荐!!!购买套餐就免费送各大视频网站会员!快速稳定、独家福利社、流媒体稳定解锁!速度快,全球上网、视频、游戏加速、独立IP均支持!基础套餐性价比很高!这里不多说,我一直正在使用,推荐购买:https://www.javaxiu.com/59919.html
weinxin
资源分享QQ群
本站是JAVA秀团队的技术分享社区, 会经常分享资源和教程; 分享的时代, 请别再沉默!
沙海
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定