测试代码:
public static void main(String[] args) {
String a = "hello2";
final String b = "hello";
String d = "hello";
String c = b + 2;
// 由于b是常量所以编译期,编译器会直接将b替换为其值,并直接拼接成"Hello2"赋值给c,这又叫做“常量传播”优化
// 故在编译期就能确定b和c的值
String e = d + 2;
// 由于赋值语句右侧使用的不同类型常量赋值,因此会使用StringBuilder实现字符串拼接
// 会先调用其append方法最后调用toString方法,而toString方法是new String并返回,即在堆中创建对象
// 故e的值需要运行时确定
String f = "hello" + "2";
// 而f的赋值语句右侧也都是字面量,或者说常量,同字符串c的解释,故在编译期就能确定f值
// 故最终的输出结果为:
System.out.println(a == c);// true
System.out.println(a == e);// false
System.out.println(a == f);// true
}
反汇编class文件输出:
C:\Users\DELL\Desktop>javap -c Solution.class
Compiled from "Solution.java"
public class LeetCodeStudy.Solution {
public LeetCodeStudy.Solution();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public static void main(java.lang.String[]);
Code:
0: ldc #2 // String hello2
2: astore_1
3: ldc #3 // String hello
5: astore_2
6: ldc #3 // String hello
8: astore_3
9: ldc #2 // String hello2
11: astore 4
13: new #4 // class java/lang/StringBuilder
16: dup
17: invokespecial #5 // Method java/lang/StringBuilder."<init>":()V
20: aload_3
21: invokevirtual #6 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
24: iconst_2
25: invokevirtual #7 // Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder;
28: invokevirtual #8 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
31: astore 5
33: ldc #2 // String hello2
35: astore 6
37: getstatic #9 // Field java/lang/System.out:Ljava/io/PrintStream;
40: aload_1
41: aload 4
43: if_acmpne 50
46: iconst_1
47: goto 51
50: iconst_0
51: invokevirtual #10 // Method java/io/PrintStream.println:(Z)V
54: getstatic #9 // Field java/lang/System.out:Ljava/io/PrintStream;
57: aload_1
58: aload 5
60: if_acmpne 67
63: iconst_1
64: goto 68
67: iconst_0
68: invokevirtual #10 // Method java/io/PrintStream.println:(Z)V
71: getstatic #9 // Field java/lang/System.out:Ljava/io/PrintStream;
74: aload_1
75: aload 6
77: if_acmpne 84
80: iconst_1
81: goto 85
84: iconst_0
85: invokevirtual #10 // Method java/io/PrintStream.println:(Z)V
88: return
}
参考文档:
String 相加解析原理String字符串相加的问题
java 字符串拼接为什么要用 StringBuilder 而不直接用 String 相加连接