Java中的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() {
    //两个字符串常量拼接,jvm直接把结果加入到常量池中。
    //如果一个字符串常量引用拼接一个字符串常量,调用的实际上是StringBuilder.append,是创建一个新的字符串
        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);
    }
  • 首先是final关键字问题。被final关键字修饰的对象,在编译的时候会被替换成字面常量。因此str10=str9+"c" 就相当于str10="ab"+"c" ,结果为true
  • 如果用字符串常量拼接成一个新的字符串,JVM会把最终的结果存储到字符串常量池中。而如果用一个字符串引用来参与拼接,则相当与调用stringBuilder.append方法,最终的对象会创建在堆中。
  • 第三个比较再次说明intern()返回的是字符串在常量池中的引用。

你可能感兴趣的:(Java中的string及string.intern())