java String 缓冲池

java 中 String 创建详细解释

字符串对象是一种特殊的对象。String 类是一个不可变的类。也就说,String 对象
一旦创建就不允许修改。
String 类有一个对应的 String 池,也就是 String pool。每一个内容相同的字符串对
象都对应于一个 pool 里的对象。

1.String创建

String s = new String("abc");  
String s1 = "abc";  
String s2 = new String("abc");  
System.out.println(s == s1);  
System.out.println(s == s2);  
System.out.println(s1 == s2);   
问题:前面三条语句分别创建了几个对象,分别是什么。后面的打印输出分别是什么?

(1)String s = new String("abc");
这句创建了两个对象,其内容都是 “abc” 。
注意:s 不是对象,只是引用。只有 new 生成的才是对象。
创建的流程是:首先括号里的 “abc” 先到 String pool 里看有没 “abc” 这个对象,没有则在 pool 里创建这个对象。所以这里就在 pool 创建了一个 “abc” 对象。然后通过new 语句又创建了一个 “abc” 对象,而这个对象是放在内存的堆里。这里的 s 指向堆里的对象。

(2)String s1 = "abc";
这条语句,s1 当然还是引用。后面的”abc”,其实就是上面括号里的”abc”,执行的是相同的操作。即 在 pool 里查找有没 “abc” 这个对象,没有则创建一个。很显然,第一条语句在 pool 里已经创建了一个”abc”,所以这条语句没有创建对象,s1 指向的是 pool 中的 “abc”。

(3)String s2 = new String("abc");
这条语句,其实和第一条是一样的。因为第一条已经在 pool 中创建了 “abc” 这个对象,所以这条语句由于在 pool 中先找到了”abc”,所以不用在 pool 中再次创建”abc”了,而只是在堆里创建了一个新的”abc”对象。s2 指向的是堆里的”abc”。注意,虽然内容都是”abc”,s 与 s2 表示的是不同的对象。

(4)下面的三个==判断(注意:==永远是判断内存地址是否相等)
s 与 s1,一个指向堆里的对象,一个指向 pool 里的,很明显是不同的对象。
s 与 s2,上面说了,虽然都是指向堆里的对象,内容也是”abc”,但是也不是相同的对象。
s1 与 s2,一个指向 pool,一个指向堆。也不是相同的对象。
所以三个都返回 false。

2.String.intern() 方法

String s = new String("abc");  
String s1 = "abc";  
String s2 = new String("abc");  
System.out.println(s == s1.intern());  
System.out.println(s == s2.intern());  
System.out.println(s1 == s2.intern()); 
问题:最后的打印输出是什么?

解答:最后的答案是 false false true

intern()方法:按照 jdk 的帮助文档来说,是返回字符串对象的规范化表示形式。通俗一点说,就是返回对应这个字符串内容的那个 pool 里的对象。
举例说明:
s1.intern()方法的执行流程是,在 pool 里去查找 s1 对应的内容(也就是”abc”)。如果找到,则返回 pool 里的对象。如果没有(老实说,我没想到有哪种情况是没有的),则在 Pool 创建这个对象,并返回。
这样就很容易理解了。s1.intern 返回的是 pool 里的”abc”对象,与 s 这个堆里的对
象肯定不同,返回 false。同理,s 与 s2.intern()也肯定不同,返回 false.第三个,s1 与s2.intern().其中 s2.intern()返回的是 pool 中的”abc”对象,而 s1 也是指向 pool 中的”abc”对象。所以返回的是 true。

3.字符串拼接

String hello = "hello";  
String hel = "hel";  
String lo = "lo";  
System.out.println(hello == "hel" + "lo");  
System.out.println(hello == "hel" + lo); 
问题:最后的打印输出是什么?

解答 true false

首先,上面已经说明了,hello hel lo 这三个都是指向 pool 中的对象。
现在我们考虑 “hel” + “lo” 按照内容来说,两个相加也就是”hello”。这个时候,这个会返回 pool 中的”hello”对象。所以,hello == “hel” + “lo” 返回的是 true 。
而”hel” + lo 虽然内容也是”hello”,但是它将在堆里面生成一个”hello”对象,并返回这个对象。所以这里的结果是 false。

总结一下就是,如果加号两边的是字面值(字面值就是直接在”“里写的值,比如上面
的”hel”与”lo”),那么将在 pool 里查找有没对应内容的对象(这里的内容就是”hello”),并返回 pool 里的对象。这和 hello 是一样的。
如果加号两边不满足上面的条件(比如,两边的值是引用值或者堆里的字符串对象)。那么将不会再 pool 里查找”hello”,而是直接在堆里生成一个新的对象。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值