关于字符串常量池,intern方法的理解

  1. 首先确定一点无论是jdk1.6还是1.7

    String c = "ab";
    

    这种字面赋值都是直接在字符串常量池生成一个字符串对象的。

  2.   String a = new String("a");
      String b = "a";
      System.out.println(a == b);
    

    输出的是false
    这说明在用new生成字符串对象的时候,只会在堆中生成字符串对象。而不会在字符串常量池生成对象,也没有引用。

  3. 明确:
    1、在jdk1.6中,intern()方法是先查找字符串常量池是否含有当前字符串,如果没有,那么就在字符串常量池中创建 该字符串,并且返回该字符串在字符串常量池中的引用。
    2、在jdk1.7中,intern()方法也是先查找字符串常量池中是否含有当前字符串,如果没有,将该字符串在堆的引用存在常量池中。并且返回该字符串在堆中的引用。
    这是因为在jdk1.6中字符串常量池在永生代(方法区)中,而在jdk1.7字符串常量池转移到了堆中。
    《深入理解Java虚拟机:JVM高级特性与最佳实践(第3版) (华章原创精品)》书中原话。
    在这里插入图片描述

上述即可解释下面的例子:
在这里插入图片描述
需要注意的是,这里看作之前字符串常量池就有java这个字符串了。

  1. 看下面示例

    String a = new String("ab");
    String b = new String("ab");
    String c = "ab";
    String d = "a" + "b";
    String e = "b";
    String f = "a" + e;
     
    System.out.println(b.intern() == a);
    System.out.println(b.intern() == c);
    System.out.println(b.intern() == d);
    System.out.println(b.intern() == f);
    System.out.println(b.intern() == a.intern());
    

    输出结果为

    false
    true
    true
    false
    true
    

    由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。

5.我们看下面示例:这个类似于我们上面在书中看到的那个例子
在这里插入图片描述
输出是false,但是按照我们上述的推理来说他应该输出的true才对。
于是我查阅各种资料发现可能是因为字符串a在常量池中已经被创建过了。尝试下面写法:

 String str1 = new String(new char[]{'a'});
 System.out.println(str1.intern() == str1);

输出的是True。
灰色的应该就是字符串常量池已经有这个字符串了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值