字符串一旦生成就不能改变。
字符串本身在堆中,而字符串常量池中有指向他的引用,字符串常量池也在堆中!
当字面量方式声明时,查找常量池,有则返回引用,没有就在堆中生成对象,同时在常量池中生成引用,如:String s = "xyz";
字面量相加,根据+的结果查找常量池,有则返回引用,没有则堆内生成对象,同时常量池中产生引用,在这里final 的变量,也看成字面量
字符串相加,如果相加时不是字面量,那么就要在堆中生成新的对象,常量池中不生成引用,String s = s1 + "abc"; --> String s = new String(s1 +"abc");
final String s1 = "a";
String s2 = "a"+"b";
String s3 = "ab"
String s4 = "ab";
System.out.println((s1+"b") == s3); true, 这里由于final string的缘故,所以a+2是两个常量,会在编译的时候直接折叠成一个字符串常量,他们都存在于字符串常量池中
System.out.println(s3 == s2); true, s2由两个常量组成,这里也折叠了
System.out.println(s3 == s4); true
String str = "a" + "b" + "c"; 根据折叠原理,这里只创建了一个对象
public class Demo {
public static void main(String[] args) {
String str1="abc"; // 使用字符串常量,从字符串常量池中取
String str2="abc";
String str3=new String("abc"); // new创建一个新对象,str3指向堆内存,str1指向常量池
String str4=new String("abc");
String str5=str4.intern(); // **intern "返回一个常量池中的固定对象。当intern方法被调用时,如果常量池中已经包含了这个String对象(用equals方法判断包含与否),那么直接返回这个对象。否则,就向常量中添加这个对象,并返回对它的引用"**
System.out.println(str1==str2);//第一行,true
System.out.println(str1==str3);//第二行,false
System.out.println(str1==str5);//第三行,true
System.out.println(str3==str4);//第四行,false
System.out.println(str3==str5);//第五行,false
System.out.println(str1.equals(str2));//第六行,true
System.out.println(str1.equals(str3));//第七行,true
System.out.println(str1.equals(str5));//第八行,true
System.out.println(str3.equals(str4));//第九行,true
System.out.println(str3.equals(str5));//第十行,true
}
}
class internDemo
{
public static void main(String[] args)
{
String s1 = new String("abc");
String s2 = s1.intern();
System.out.println(s1 == s2); // return false
}
}