我们看这样一行简单的字符串赋值操作的Java代码。
String a = “i042416”;
使用命令行将包含了这行代码的Java类反编译查看其字节码:
javap -v constant.ConstantFolding
ldc #16: 首先JVM底层的原生方法StringTable::intern被调用,生成String的内部存储实现char[]。然后执行ldc #16, 将常量池内的代号为#16的常量加载到栈上,即i042416。
2. astore_1:将"i042416"的引用存储到序号为1的本地变量中(即我们代码中的局部变量a)。
下面再看一个稍微复杂一点的例子。
做一个字符串拼接的操作。
String aa1 = “i042416”;
String aa2 = “jerrywang”;
String aa3 = “i042416” + “jerrywang”;
可以看到,在编译阶段,编译器字节把两个字符串常量的值进行拼接,结果"i042416jerrywang"存储到变量aa3中, 作为一个新的字符串常量, 在常量池中代号为#21。
因此,变量aa1和aa3实际上指向常量池中的同一个常量,因此直接用==比较也返回true。
String aa1 = “i042416jerrywang”;
String aa2 = “jerrywang”;
String aa3 = “i042416” + “jerrywang”;
System.out.println(aa1 == aa3);