堆:存放所有new出来的对象。
常量池:存放字符串常量和基本类型常量(public static final)。
对于字符串:其对象的引用都是存储在栈中的,如果是编译期已经创建好(直接用双引号定义的)的就存储在常量池中,如果是运行期(new出来的)才能确定的就存储在堆中。
String s1 = "china"; //编译期创建,存储在常量池中
String s2 = "china";
String ss1 = new String("china"); //运行期new出来,存储在堆中
String ss2 = new String("china");
if (s1 == s2)//true
{
System.out.println("s1 == s2");
} else {
System.out.println("s1 != s2");
}
if(ss1 == ss2) {//false
System.out.println("ss1 == ss2");
} else {
System.out.println("ss1 != ss2");
}
if (s1.equals(s2))//true
{
System.out.println("s1.equals(s2)");
} else {
System.out.println("s1 not equals s2");
}
if (ss1.equals(ss2))//true
{
System.out.println("ss1.equals(ss2)");
} else {
System.out.println("ss1 not equals ss2");
}
if (s1 == ss1)//false
{
System.out.println("s1 == ss1");
} else {
System.out.println("s1 != ss1");
}
if (s1.equals(ss1))//true
{
System.out.println("s1.equals(ss1)");
} else {
System.out.println("s1 not equals ss1");
}
结果:
s1 == s2
ss1 != ss2
s1.equals(s2)
ss1.equals(ss2)
s1 != ss1
s1.equals(ss1)
对于equals相等的字符串,在常量池中永远只有一份,在堆中有多份。
对于通过new产生一个字符串(假设为”china”)时,会先去常量池中查找是否已经有了”china”对象,如果没有则在常量池中创建一个此字符串对 象,然后在堆中再创建一个常量池中此”china”对象的拷贝对象。
有道面试题:String s = new String(“xyz”);产生几个对象?一个或两个,如果常量池中原来没有”xyz”,就是两个。