StringTable字符串的拼接,intern()工作原理

1.编译期的优化 

public class Test {
    public static void main(String[] args){
        String s1="a";
        String s2="b";
        String s3="ab";
        String s4="a"+"b";//在编译时就能确认为"ab"
        String s5=s1+s2;//new StringBuffer().append("a").append("b").toString
        System.out.println(s3==s4);//true
        System.out.println(s3==s5);//false
}
1.String s4="a"+"b";
由于"a","b"在编译时系统可以肯定其结果为"ab",所以在加载进内存时,直接寻找常量池是否存在"ab",如果有直接指向这块地址
2.String s5=s1+s2;
有一个变量s2,无法确定,它会调用StringBuffer.append("a"),append("b")添加元素。而StringBuffer的toString方法返回的是:return new String("ab");由此可见生成了一个新的引用类型对象,引用类型对象存储在堆中。所以s5指向的是堆中的地址,s3指向常量池中地址

2.intern()方法

首先要知道intern()在jvm中是如何工作的。如s.intern(),先看常量池中是否有与s相同的值,如果没有,就将s加入常量池并返回其对象的引用;如果有则不会将s加入常量池,但还是会返回其常量池中 对象的引用(与s相同的值)。

String s1=new String("ab");
String s2=s1.intern();
System.out.println(s1==s2);//false

先看s1,会创建2个对象,一个"ab"对象在常量池中,对象new String()在堆中。

执行s1.intern()方法时,在常量池中已经有"ab"了,所以不会在将new String("ab")的"ab"加入常量池,最后将常量池中对象“ab”的地址返回给s2。s1指向的是堆中地址,而s2指向的是常量池中地址,所以s1==s2为false。

再看另一种情况 

String s1=new String("a")+new String("b");//new String("ab")
String s2=s1.intern();
System.out.println(s1==s2);//true

 new String("ab")是在运行时产生的,存在于堆中,在常量池中还没有"ab"对象。接下来执行s1.intern(),将"ab"放入常量池并将其地址返回给s2和s1。所以s1==s2为true。

 

 再看另一种情况

String s1=new String("a")+new String("b");//new String("ab")
s1.intern();
String s2="ab";
System.out.println(s1==s2);//true

此时s2="ab"去常量池寻找是否有"ab",发现有,所以此时s2指向的是常量池中的"ab",与s1的指向同一地址,所以s1==s2结果为true。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

IABQL

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值