美文网首页
Java中的string及string.intern()

Java中的string及string.intern()

作者: 柳蒿 | 来源:发表于2019-08-21 23:22 被阅读0次

    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()

          本文链接:https://www.haomeiwen.com/subject/fsbwsctx.html