如果常量池存在“abc”常量
变量s1指向常量池的"abc"
如果常量池不存在"abc"常量
常量池先创建"abc",然后变量s1指向常量池的"abc"
如果常量池存在“abc”常量
将“abc”拷贝到Java堆,变量s1指向Java堆的"abc"
如果常量池不存在"abc"常量
常量池先创建"abc",然后将“abc”拷贝到Java堆,变量s1指向Java堆的"abc"
问题
public static void main(String[] args) {
String s1 = "hello";
String s2 = new StringBuilder("he").append("llo").toString();
String s3 = s2.intern();
String s4 = "hello";
System.out.println(s1 == s2); // false
System.out.println(s2 == s3); // false
System.out.println(s3 == s4); // true
}
分析
s1 = "hello",将"hello"放入方法区,并且s1指向"hello"。s2获取到了“hello”,但是这并不是从方法区获取的"hello",
s3=s2.intern()是将s2的"hello"与方法区进行对比,如果存在"hello",那么就返回原来"hello"的方法区地址,
否则放入方法区并返回新的方法区地址。这里因为方法区已经有"hello",所以返回了以前的地址,所以s2 != s3。
s4 = "hello",指向了方法区的"hello"的地址。
再看一下下面的例子
String s2 = new StringBuilder("he").append("llo").toString();
String s3 = s2.intern();
String s4 = "hello";
System.out.println(s2 == s3); // true
System.out.println(s3 == s4); // true