一道老生常谈的字符串对象创建问题

问题和结论

老生常谈的一个问题,String s = new String(“abc”)创建了几个对象,String s = "abc"又是创建了几个对象?

先说结论:

String s = new String(“abc”)创建了1个或2个对象,String s = "abc"创建了1个或0个对象

接下来我们探究一下String s = new String(“abc”) 和String s = "abc"创建过程的区别。

String s = new String(“abc”)的创建过程

  • 系统先在字符串常量池里面寻找是否有一个"abc"的字符串,
  • 如果有的话,则在堆中复制一个该字符串,并且将堆中的引用赋给s,这个时候系统只创建了一个对象,即堆中的对象;
  • 如果没有的话,则会先在字符串常量池中先创建一个字符串为"abc"的常量,然后再复制到堆里面,最后将堆所在的地址赋给s,这个时候创建了两个对象;

String s = "abc"的创建过程

  • 系统先在字符串中寻找是否存在"abc"的常量
  • 如果存在,则直接将该"abc"在常量池中的地址赋给s,这个时候,系统没有创建新对象。
  • 如果不存在,则在常量池中新建一个"abc"并放入常量池里面,然后再返回该地址,这个时候,系统创建了一个对象。

示例

String s = "abc";
System.out.println(s == "abc"); // true

因为s 和 “abc” 都是指常量池里面"abc" 的地址,所以true;

String s = new String("abc");
System.out.println(s == "abc"); // false

因为s 指向的是堆里面新建的对象的地址,而"abc"指向的是常量池里面的地址,所以不等。

String s = new String("abc");
String s1 = new String("abc"); 
System.out.println(s == s1); // false

s和s1都是在堆中新建了不同的对象,虽然内容一样,但是两则是位于堆中不同地址的空间,所以是不相等的。

String s = "a" + "b"; 
System.out.println(s == "ab"); // true

此处虚拟机会做优化,会在常量池里面寻找"a" + “b” 结果后的字符串即"ab",所以两者都是对应常量池中"ab"的地址。

String s1 = "a";
String s2 = "b";
String str = s1 + s2;
System.out.println(str == "ab"); // false

虽然s1和s2各自指的是常量池里面"a","b"的引用,但是string在做加法或者subString、replace等方法的时候,实际上返回的是new String()的结果,因此str指向的是堆中的地址,所以不相等。

String s = "abc";
String ss = new String("abc").intern();
System.out.println(s == ss); // true

虽然此处ss是new出来的新对象,但是由于调用了intern方法,这个方法会返回常量池中相等值的字符串的地址,所以最后ss指向的是常量池中"abc"的地址,所以相等。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值