1. 结论
先说结论,创建了2个对象,为什么网上有的地方说创建了一个?那是因为这种情况下new String(“abc”)的确创建了一个对象:
String a = "abc";
String b = new String("abc");
下面从从几道面试题由浅入深的来解释一下
注意,以下环境均基于JDK1.8,其他版本如1.7略有不同
2. String a = “abc”
先看一下生成的字节码
0 ldc #2 <abc>
2 astore_1
3 return
0 ldc
查找后面索引为 #2 对应的项, #2 表示常量在常量池中的位置。在这个过程中,如果发现 StringTable 已经有了内容匹配的 String 引用,则直接返回这个引用,反之如果 StringTable 里没有内容匹配的 String 对象的引用,则会在堆里创建一个对应内容的 String 对象,然后在 StringTable 驻留这个对象引用,并返回这个引用,之后再压入操作数栈中2 astore_1
弹出栈顶元素,并将栈顶引|用类型值保存到局部变量 1 中,也就是保存到变量S中3 return
执行 void 函数返回
所以,在这种情况下,只在堆中创建一个String对象,图释如下:
首先明确一点,堆中的StringTable其实是数组+链表的形式
![image-20221030003149814](https://markdown-pic-june.oss-cn-beijing.aliyuncs.com/2022/10/30/image20221030003149814.png)
下面用IDEA强大的load classes功能验证下:
![image-20221030000931011](https://markdown-pic-june.oss-cn-beijing.aliyuncs.com/2022/10/30/image20221030000931011.png)
运行至第一个断点处:
![image-20221030000849116](https://markdown-pic-june.oss-cn-beijing.aliyuncs.com/2022/10/30/image20221030000849116.png)
运行至第二个断点处:
![image-20221030000916721](https://markdown-pic-june.oss-cn-beijing.aliyuncs.com/2022/10/30/image20221030000916721.png)
至此,一般可以说这种方式创建了1个对象。另外,char[]数组其实是特殊的对象,如果你把它包含在内的话那就是2个,但是这个不是这个问题的重点,可以忽略这一点。
3. String b = new String(“abc”)
先上字节码:
0 new #2 <java/lang/String>
3 dup