String创建的两种方式在JVM层面的区别

String 常见的创建的两种方式

  • 直接赋值的方式:会先去字符串常量池中查找是否已经有此值,如果有则把引用地址直接指向此值,否则会先在常量池中创建,然后再把引用指向此值;

  • new String()的方式:一定会先在堆上创建一个字符串对象,然后再去常量池中查询此字符串的值是否已经存在,如果不存在会先在常量池中创建此字符串,然后把引用的值指向此字符串,如下代码所示:

String s1 = new String("Java");

String s2 = s1.intern();

String s3 = "Java";

System.out.println(s1 == s2); // false

System.out.println(s2 == s3); // true

它们在 JVM 存储的位置,如下图所示:
image

小贴士:JDK
1.7 之后把永生代换成的元空间,把字符串常量池从方法区移到了 Java 堆上。

除此之外编译器还会对 String 字符串做一些优化,例如以下代码:

String s1 = "Ja" + "va";

String s2 = "Java";

System.out.println(s1 == s2);

虽然 s1 拼接了多个字符串,但对比的结果却是 true,我们使用反编译工具,看到的结果如下:

Compiled from "StringExample.java"

public class com.lagou.interview.StringExample {

  public com.lagou.interview.StringExample();

    Code:

       0: aload_0

       1: invokespecial \#1                  // Method java/lang/Object."\<init\>":()V

       4: return

    LineNumberTable:

      line 3: 0

  public static void main(java.lang.String[]);

    Code:

       0: ldc           \#2                  // String Java

       2: astore_1

       3: ldc           \#2                  // String Java

       5: astore_2

       6: getstatic     \#3                  // Field java/lang/System.out:Ljava/io/PrintStream;

       9: aload_1

      10: aload_2

      11: if_acmpne     18

      14: iconst_1

      15: goto          19

      18: iconst_0

      19: invokevirtual \#4                  // Method java/io/PrintStream.println:(Z)V

      22: return

    LineNumberTable:

      line 5: 0

      line 6: 3

      line 7: 6

      line 8: 22

}

从编译代码 #2 可以看出,代码 "Ja"+"va" 被直接编译成了 "Java" ,因此 s1==s2
的结果才是 true,这就是编译器对字符串优化的结果。

posted @ 2020-08-13 12:08  0小豆0  阅读(244)  评论(0编辑  收藏  举报
隐藏
对话
对话