JVM-intern()

本文详细解析了Java中的String类的intern()方法在JDK7前后的行为变化。介绍了当使用new String()创建对象并调用intern()时,如何处理字符串常量池和堆中的对象,以及特殊情况下的字节码行为。还给出了面试题的相关解答,分析了不同JDK版本下intern()方法对对象引用的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在这里插入图片描述
intern()的理解1:
由于jdk7之前,字符串常量池是在永久代里面的(即方法区),新创建出来的两个对象一个在堆中,一个在永久代中
intern()在jdk7之后,字符串常量池放在了堆里面,此时如果如果new String().intern(),由于堆中保存两个相同的对象很浪费,所以调用了intern()之后,只有一个对象实体,并且在字符串常量池中。
intern()的理解2:
如果在调用intern()之前,字符串常量池和堆区已经生成了对象,则再次调用intern()也不会消除原本堆区的对象。
在这里插入图片描述
答案:2个

  1. 第一个是由于new String(),在堆区创建了一个对象,这个对象是字符串常量池中的对象的地址
  2. 同时在字符串常量池中也创建了一个对象,这个对象就是ab实体
    说的更加具体一点:
    由于是new String(),首先会在堆区创建一个ab的对象,然后就会看字符串常量池中是否有这个对象,如果有的话,就不在字符串常量池中新创建对象,直接将这个堆区的对象指向字符串常量池中的对象即可;如果没有的话,就创建这个ab的对象,并且将这个新创建出来的对象的地址,返回给堆区这个对象。在这里插入图片描述

拓展答案:
在这里插入图片描述
注意:
针对于new String()的一个特例!!!
这里通过看字节码文件可以看到,如果是StringBuilder中也调用了new String(),此时并没有这个StringBuilder中的 new String(),并没有在字符串常量池中创建新的对象(特例,记住即可)

面试题:

在这里插入图片描述
第一部分:
jdk 6 为false的原因:
s为堆空间的变量,由intern()了之后,1加载进了字符串常量池,但是s指向的是堆空间的对象,而s2此时指向的是字符串常量池中的对象,所以不一样;

jdk 7/8 为false的原因:
理由同上

第二部分:
String s3 = new String(1)+new String(1);
此时并没有在字符串常量池中创建对象,仅仅在堆区创建了一个对象,通过intern()之后,将11也加到了字符串常量池中
不同的是:
jdk 6:
由于此时的字符串常量池没有在堆中,而s3指的是堆区的对象,s4是字符串常量池中的对象,是false;
jdk 7/8:
由于intern()的特点,导致此时的堆中只会存在一个对象,所以s3和s4是同一个对象。
总结:
在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值