string及string.intern()相关

Java中的string类型一直是一个热点问题,也是Java面试问题中的常客。string类型涉及到常量池,堆等方面,是考察Java基础的很好的问题。我整理了一些常见的情况,做了一个总结。

  public void run1() {
       String str1 = "abc";
       String str2 = new String("abc");
       System.out.println(str1 == str2);
   }

这是最基本,也是最常见的问题了。创建str1时,JVM把字面值常量 "abc"加入到位于堆的字符串常量池中,然后返回这个常量的引用,str1就是这个引用。创建str2时,JVM在堆中开辟一小块空间,用来容纳 “abc”,然后返回这一小块空间的首地址,str2就是这个地址。

    public void run2() {
        //intern的作用,返回字符串在常量池中的引用,如果不能再常量池中找到该字符串,则把该字符串加到
        String str1 = "abc";
        String str2 = new String("abc");
        String str3 = new String("abc");

        System.out.println(str2 == str3);
        System.out.println(str2.intern() == str3.intern());
        System.out.println(str2 == str3.intern());
        System.out.println(str1 == str2.intern());
    }
  • str2,str3是两个单独new出来的对象,在堆中各自占据一定的空间,因此str2==str3为false。
  • intern()的作用是返回string类型对象在字符串常量池中的引用,如果这个string类型不在字符串常量池中,则把这个字符串添加到字符串常量池中,然后在返回这个在常量池中的字符串的引用。因此,虽然str2,str3在堆中占据不同的内存空间,但是调用intern()方法后,两个字符串都被复制到字符串常量中,而对于相同的字符串对象,字符串常量池又只会保留一个,因此str2.intern() 返回的结果就和str3.intern()相同
  • 按照上面一点,str3.intern()返回的是str3在字符串常量池中的引用。因此str2 == str3.intern()为false
  • 根据第二点,str1是常量池中"abc"的引用,str2.intern()最终返回的也是"abc"在常量池中的引用,因此str1 == str2.intern()为true
    public void run3() {
        String str1 = "abc";
        String str2 = new String("abc");
        str2.intern();
        System.out.println(str1 == str2);
    }

网上对intern()的解释大多都是会把字符串从堆中添加到字符串常量池中。但是先执行intern(),然后再比较str1 == str2,结果却是false。因为str2仍旧是JVM在堆中给"abc"对象所分配的内存块的首地址。

    public void run4() {
        String str1 = "abc";
        String str4 = "ab";
        String str5 = "ab" + "c";
        String str6 = str4 + "c";

        //final变量在编译后会直接替换成对应的值
        final String str7 = str4;
        String str8 = str7 + "c";

        final String str9 = "ab";
        String str10 = str9 + "c";

        System.out.println(str1 == str5);
        System.out.println(str1 == str6);
        System.out.println(str1 == str6.intern());
        System.out.println(str1 == str8);
        System.out.println(str1 == str10);
    }
  • 如果用字符串常量拼接成一个新的字符串,JVM会把最终的结果存储到字符串常量池中。而如果用一个字符串引用来参与拼接,则相当于调用stringBuilder.append方法,最终的对象会创建在堆中。因此str1 == str6为false
  • 第三个比较再次说明intern()返回的是字符串在常量池中的引用。
  • 首先是final关键字问题。被final关键字修饰的对象,在编译的时候会被替换成字面常量。因此str10=str9+“c” 就相当于str10=“ab”+“c” ,结果为true
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值