JDK1.6和JDK1.7关于String.intern()方法差异详解

JDK1.6和JDK1.7关于String.intern()方法差异详解

最近看《深入理解Java虚拟机》时,在2.4.3 方法区和运行时常量池溢出这一章节看到了String.intern(),当时还没太明白,网上查一些资料,之后再仔细揣摩书本时才算是有了一些自己的理解,深知笔记对于理解知识的重要性,于是写下这篇博客,也算是自己成长路上的一个记录。

前置知识

因为使用的是《深入理解Java虚拟机》书中的例子,里面涉及对象创建,因此准备先理清Java创建字符串对象的过程。

public class Helloworld {
    public static void main(String[] args) {
        String s1=new String("abc");
        String s2="abc";
        String s3="abc";
        String s4=new String("abc");
        System.out.println(s1==s2);
        System.out.println(s1==s4);
        System.out.println(s2==s3);
        System.out.println(s2==s4);
    }
}
结果分别是:
false
false
true
false

new对象的过程如下String s1=new String("abc");

首先在字符串常量池中查找是否存在字符串“abc”,如果不存在,那么先在字符串常量池中开辟一块内存,用于存放“abc”,然后在堆内存中开辟一块空间,将字符串常量池中的“abc”复制一份到堆空间的存储区,最后栈空间中的变量s1指向堆空间的字符串对象。一共创建了两个对象

String.intern()作用

jdk1.6

如果字符串常量池中存在对应字符串对象,那么直接返回该对象的引用,如果不存在,先将该字符串添加到常量池中,然后再返回该对象的引用。

JDK1.6

jdk1.7

如果字符串常量池中存在对应字符串对象,那么直接返回该对象的引用,如果不存在,则会在字符串常量池中开辟一块内存区域,存放堆内存中该对象的引用(不会再复制),然后返回该引用。

JDK1.7

String.intern()测试代码

public class intern {
    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结果

false
false

String str1 = new StringBuilder("计算机").append("软件").toString();

执行结束时,堆中有“计算机”、“软件”、“计算机软件”三个字符串对象,字符串常量池有“计算机”、“软件”两个字符串对象。

str1.intern()

将堆中的“计算机软件”复制到字符串常量池中,并返回字符串常量池中该对象的引用。

str1指向堆内存,故而返回false


str2分析同理!

jdk1.7结果

false
true

String str1 = new StringBuilder("计算机").append("软件").toString();

执行结束时,堆中有“计算机”、“软件”、“计算机软件”三个字符串对象,字符串常量池有“计算机”、“软件”两个字符串对象。

str1.intern()

在字符串常量池中创建一个指向堆中“计算机软件”的引用,并返回。

str1指向堆内存,故而返回true


str2书中给出的原因是

因为“java”这个字符串在执行StringBuilder.toString()之前已经出现过,字符串常量池中已经有它的引用,不符合“首次出现”原则,因此返回false。

但是我自己这块没有明白为啥字符串常量池为什么有引用,于是我把字符串改了下

String str2 = new StringBuilder("jaa").append("va").toString();
System.out.println(str2.intern() == str2);

得到结果为True

然后再改

String str3 = "jaava";
String str2 = new StringBuilder("jaa").append("va").toString();
System.out.println(str2.intern() == str2);

结果为false,这样就也能反映出是符合上述策略的。

参考文献

1.深入理解Java虚拟机第二版, 周志明

2.String.intern()方法

3.String str=new String(“abc”);创建几个对象?

4.new创建字符串对象过程

5.String.intern() 方法有什么作用?

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值