new String(“abc“)到底创建了几个对象

本文探讨了Java中创建String对象的不同方式,包括`String a = "abc"`和`String b = new String("abc")`,揭示了这两种方式分别创建的对象数量。在JDK1.8环境下,`String a = "abc"`只创建一个对象,而`String b = new String("abc")`创建了两个。此外,文章还讲解了`intern()`方法的工作原理以及字符串常量拼接的优化策略。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

下面用IDEA强大的load classes功能验证下:

image-20221030000931011

运行至第一个断点处:

image-20221030000849116

运行至第二个断点处:

image-20221030000916721

至此,一般可以说这种方式创建了1个对象。另外,char[]数组其实是特殊的对象,如果你把它包含在内的话那就是2个,但是这个不是这个问题的重点,可以忽略这一点。

3. String b = new String(“abc”)

先上字节码:

 0 new #2 <java/lang/String>
 3 dup
 4 ldc #3 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值