String字符串的常量池
public class Demo10 {
public static void main(String[] args) {
String s1 = "Programming";
String s2 = new String("Programming");
String s3 = "Program";
String s4 = "ming";
String s5 = "Program" + "ming";
String s6 = s3 + s4;
//s1创建于字符串缓冲池,而s2创建于内存(堆)中,尽管二者内容都是"Programming",但地址不在一个区,故比较地址的结果为 false。
System.out.println(s1 == s2);//false
/*
String s5 = "Program" + "ming";的操作, 由于表达式的两个算子都是字符串常量,所以操作始终在池内进行。
在池内找到 "Program" 和 "ming"之后,将两者连接成 "Programming", 然后,又在池内找到了相同值的对象s1,
于是就将这个地址,赋予 s5。因此,比较s1 和 s5 的地址,当然相同,结果为 true。
*/
System.out.println(s1 == s5);//true
/*
用操作符 new 创建的字符串对象, 如 String s2 = new String("Programming");,
或者有字符串变量、操作符 '+' 参与的表达式,如 String s6 = s3 + s4;
这样得到的引用,都要放到内存(堆heap)中保存。故,在堆中,若有几个值相同的对象,它们分别都会有各自的地址。
s6的地址,存在堆里。而s1的地址存在池里。尽管两者值相同,但地址一定不同,结果为 false。
*/
System.out.println(s1 == s6);//false
/*
因为s6调用了它的成员方法inner(),结果返回了一个在池中的、具有相同值“Programming”的对象s1的地址。
用它和s1的地址相比较,当然相等,结果为 true。
*/
System.out.println(s1 == s6.intern());//true
// s2的地址在堆里,s2.inner() 的地址在池里。一个在堆中,一个在池内,二者比较,肯定不等,结果为false。
System.out.println(s2 == s2.intern());//false
/*
String对象的intern方法会得到字符串对象在常量池中对应的版本的引用
(如果常量池中有一个字符串与String对象的equals结果是true),
如果常量池中没有对应的字符串,则该字符串将被添加到常量池中,然后返回常量池中字符串的引用;
*/
}
}