一道代码分析题浅析String的intern()方法

7 篇文章 0 订阅

最近在阅读《深入理解Java虚拟机》,书中在介绍方法区和运行时常量池溢出时使用了String.intern()进行测试。

String.intern()是一个Native方法,它的作用是:如果字符串常量池中已经包含了一个等于此String对象的字符串,则返回代表池中这个字符串的String对象;否则,将此String对象包含的字符串添加到常量池中,并且返回此String对象的引用。

但是JDK1.7之后String.intern()方法发生了改变,以下面这段代码(书上的代码)为例介绍String.intern()方法于jdk1.6和jdk.1.7执行结果的不同。

public static void main(String[] args) {
	String str1 = new StringBuilder("计算机").append("软件").toString();
	System.out.println(str1.intern() == str1);
	
	String str2 = new StringBuilder("ja").append("va").toString();
	System.out.println(str2.intern() == str2);
}

 jdk1.6执行结果:

jdk1.7执行结果:

产生差异的原因:

① 在JDK1.6中,intern()方法会把首次遇到的字符串实例复制到永久代(JDK 1.6及之前的版本中字符串常量池分配在永久代内,而JDK 1.7之后已经把原本放在永久代的字符串常量池移出),且intern()方法返回的也是永久代中的这个字符串实例的引用,而由StringBuilder创建的字符串实例在Java堆上,所以两者必然不是同一个引用,两个比较必然返回false.

 ② JDK 1.7的intern()实现不会再复制实例,只是在常量池中记录首次出现的实例引用,因此intern()返回的引用和由StringBuilder创建的那个字符串实例是同一个。

有的同学肯定会有疑问,为什么JDK 1.7中第二个比较会输出false?

答:“java”这个字符串比较特殊,它是固定存在字符串常量池中的(这个知识点真的很坑)。所以这时候执行intern()方法发现字符串常量池中已经存在“java”,所以不满足“首次出现”的原则,因此返回true.

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值