String a = "ab";
String b = "ab";
System.out.println(a == b);// true
b = a;
a += "cd";
System.out.println(a == b);// false
System.out.println("a=" + a + " " + "b=" + b);// a=abcd b=ab
String c = new String("ab");
String d = new String("ab");
System.out.println(c == b);// false
System.out.println(c == d);// false
String e = "a" + "b";
System.out.println(b == e);// true
理解上面的代码的结果前我们要知道常量池中的字符串是不会重复的,常量池中的字符串是不能改写的,“==”,比较的是引用变量其在栈内存的内容,即变量指向的字符串常量实例内存的首地址。
String a=“ab” 当执行这个语句时,JVM会检查常量池中有没有"ab",如果有,a就会存储“ab”在常量池中的首地址;如果没有,则会在常量池中创建一个“ab”并在a中存储“ab”的首地址。
因为常量池中的字符串是不会重复的,所以b会和a指向同一个“ab”,两者存储的地址相同
如果对字符串进行改写a+=“cd”,此时常量池中的字符串是不会被改写的,他会开辟一个新地址,存储“abcd”,再将这个常量的地址赋给a
我们知道我们创建一个对象时(如:new String(“ab”))会在堆区开辟一个空间来存储这个对象,每个不同的对象在堆中的存储空间都是不同的,所以c,d中存储的是不同的对象地址,但是两个对象中存储的是常量池中的同一个“ab”
当进行“a”+“b”运算时,由于常量池中没有"a","b"所以会在常量池中临时开辟两个空间分别存储“a”,“b"进行运算,运算后得到的值会在常量池中检查是否存在,这里“ab”已经存在了,所以e存储的地址与b中的相同;如果不存在就会在常量池中开辟新的空间存储。