Java String intern()原理及其底层相关

在看过多篇文章后的个人理解,有错请指正!   【侵删】

参考:
关于String底层使用的是char数组还是byte数组以及一点String面试问题
字节码层面解析String到底创建了几个对象以及String扩展之intern()方法
等。

  1. 程序中只有直接写上双引号字符串,才在字符串常量池中
  2. 常量池在1.7之后,放置在了堆空间之中。
  3. String类中对象两种实例化的区别:
    1)直接赋值只会开辟一块堆内存空间,且字符串对象可以保存在对象池中以供下次使用;
    2)采用构造方法会开辟两块堆内存空间,使用intern()方法后可以手工入池。

案例1

String str = "abc";
String str2 = new String("abc");
System.out.println(str == str2); // false 地址空间的比较
System.out.println(str.equals(str2)); // true
str2 = str2.intern();
System.out.println(str == str2); // true
String str3 = "abc";
System.out.println(str == str3); // true
String str4 = new String(new char[]{'a','b','c'});
System.out.println(str == str4); // false
System.out.println(str4 == str2); // false

在这里插入图片描述

当new String(“abc”)时,在常量池中会创建一个“abc”常量,然后再使用该值对堆中对象初始化。所以不仅是String a = “abc”,直接赋值时会在常量池中产生“abc”字符串。

在这里插入图片描述
当intern()方法被调用的时候,如果在字符串常量池中已经包含了这个String对象,那么就返回该String对象。否则这个String对象就会被添加到常量池中,然后返回该对象的一个引用。并且保证常量池中不会重复。

案例2

String s0="java No.1";
String s1="java ";
String s2="No.1";
String s3="java "+"No.1";
String s4=s1+"No.1";
String s5=s1+s2;
System.out.println(s3==s0);
System.out.println(s4==s0);
System.out.println(s5==s0);
System.out.println(s5==s4);
System.out.println("-----------------------------");
final String s6="I am ";
final String s7="guYue";
String s8="I am guYue";
String s9=s6+s7;
System.out.println(s8==s9);

结果如下:

true
false
false
false
--------------------------
true

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

我们知道对象s调用intern()方法时,字符串常量池中并没有对象"ab",所以我们就需要执行将"ab"添加到字符串常量池的的操作。而这时在不同的jdk版本可能就有不同的操作。
①jdk6及之前在常量池中创建一个String对象并返回该对象的引用。
②jdk7及之后在常量池中保存常量池外String对象的引用,并返回该引用。
在这里插入图片描述
总结:
① “?”+"?"
底层直接优化为“??”

String s0="java No.1";
String s1="java ";
String s2="No.1";
String s3="java "+"No.1";

查看字节码会发现s0和s3加载过程完全相同。所以是同一个对象

②“?”+si ,si+"?" , si+sj
我把这三种归为一类因为都有引用类型加入运算。
这时候底层会new一个StringBuilder再调用append方法,最后调用toString方法完成拼接。

String s4=s1+"No.1";

查看字节码会发现与上述过程一致。最后再看一下StringBuilder的toString()方法源码,可以发现它返回的是new String,所以肯定不是同一个对象

//StringBuilder的toString()方法
@Override
public String toString() {
    // Create a copy, don't share the array
    return new String(value, 0, count);
}

③final s1+final s2
这是两个final修饰的String引用的拼接。

final String s6="I am ";
final String s7="guYue";
String s8="I am guYue";
String s9=s6+s7;

查看代码的字节码,会发现 最后两行的字节码操作过程是完全相同的,即直接从字符串常量池中取。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值