JDK8 及之后的版本, 字符串常量池是在堆中,尽管通过new 关键字创建的对象会保存在堆中,以例子记录。
详细说明在注释中,包含字符串拼接(使用了StringTable)、String 的intern() 方法等。
最好是查看反编译之后的字节码指令,可以清晰看到执行的过程(尤其是使用 + 进行字符串拼接的时候,会创建一个StringBuilder 对象,并且不会把结果放入字符串常量池中-字节码指令中没有ldc )
public class StringTableTest {
public static void main(String[] args) {
String s1 = "a";
String s2 = "b";
String s3 = "ab"; //s3 局部变量, 指向堆中的String对象,该对象的内容为引用, 指向字符串常量池的"ab" (堆空间)
//字符串拼接,会先创建一个StringBuilder对象, 再调用append拼接
String s4 = s1 + s2; //s4 局部变量, 指向堆中的String对象,该对象的内容即为"ab"
String s5 = new StringBuilder("ab").toString();//s5 局部变量, 指向堆中的String对象,该对象的内容即为"ab"
//intern 操作(native方法): 如果字符串常量池中有改字符串,则返回地址;否则把字符串放入字符串常量池中并返回该地址
String s6 = s4.intern(); //!!!
String s7 = s3.intern();
//s3 s4 s4 是不同的String对象引用
System.out.println("s3==s4 " + (s3 == s4)); //false
System.out.println("s3==s5 " + (s3 == s5)); //false
System.out.println("s4==s5 " + (s4 == s5)); //false
System.out.println("s4==s6 " + (s4 == s6)); //false --> s4 是指向堆中的String 对象, s6 为指向字符串常量池中的对象 "ab"
System.out.println("s3==s6 " + (s3 == s6)); //true
System.out.println("s3==s7 " + (s3==s7)); //true
}
}

本文深入探讨了JDK8中字符串常量池的位置变化及其对字符串操作的影响,包括字符串拼接、StringBuilder的使用及intern方法的作用原理。通过具体示例分析了不同字符串操作在内存中的行为差异。
2777

被折叠的 条评论
为什么被折叠?



