JVM 之 String 引用机制解析:常量池、堆内存与 intern 方法

关于常量池中的String类型的数据,在JDK6中只可能是对象,在JDK7中既可以是对象也可以是引用

案例一:

String s1 = new String("1");
String s2 = "1";
System.out.println(s1 == s2);
  • s1: 执行 new String("1"),JVM 首先在字符串常量池中查找或添加字面量 "1",然后在堆内存中新建一个内容为 "1"String 对象。s1 指向的是这个堆对象。
  • s2: 指向字符串常量池中已有的 "1"
  • 结果: s1 == s2 比较的是引用地址,一个在堆中、一个在常量池中,因此结果为 false

案例二:

String s1 = new String("1") + new String("1");
String s2 = "11";
System.out.println(s1 == s2);
  • s1: 通过两个 new String("1") 创建两个堆中对象,然后使用 StringBuilder 进行拼接,最终生成了一个新的 "11" 字符串对象,存放在堆内存中
  • s2: "11" 是字面量,编译器会将其放入字符串常量池,并由 s2 指向。
  • 结果: s1 指向堆中的 "11" 对象,s2 指向常量池中的 "11",引用不同,结果为 false

案例三:

String s1 = new String("1") + new String("1");
String s2 = s1.intern();
System.out.println(s1 == s2);
  • s1: 与案例二一样,通过拼接在堆中创建了一个 "11"String 对象。
  • intern(): 在 JDK7+ 中,intern() 会将当前堆中的对象引用尝试放入字符串常量池中。如果常量池中尚未存在 "11",则将该对象引用添加进去,并返回这个引用。
  • s2: 返回的是字符串常量池中的引用,而由于常量池中原本没有 "11",此时存入的是 s1 指向的堆对象本身。
  • 结果: s1 == s2,两者引用相同,结果为 true

JVM冷启动时,String str = “1”

执行流程:

  • JVM 首先会去常量池中查找有没有字面量 "1"

    • 若没有,JVM 会在堆中创建一个字符串对象 "1",并将它的引用记录进常量池;
    • 若已存在,JVM 直接复用已有对象的引用。
  • 局部变量 str 会从常量池中取出引用,指向对应的字符串对象。

所以即使没有显式地写 newJVM 也可能创建对象,但前提是该字面量还未进入常量池。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值