看代码:
String s1 = "helloworld";
final String s2 = "hello";
String s3 = "hello";
String s4 = s2 + "world";
String s5 = s3 + "world";
System.out.println(s1 == s4); //true
System.out.println(s1 == s5); //flase
String a = new String("helloworld");
final String b = new String("hello");
String c = new String("hello");
String d = b + "world";
String e = c + "world";
System.out.println(a == d); //false
System.out.println(a == e); //flase
从中可以看出直接赋值和new是不一样的。
在java中,字符串其对象的引用都是存储在栈中的,如果编译期已经创建好(直接双引号定义的)就存储在常量池中,如果是运行期(new出来的)才能确定的就存储在堆中。对于equals相等的字符串,在常量池中永远只有一份,在堆中可以有多份。
对于上面代码,直接赋值(String s1 = "helloworld"),可能创建一个对象或者不创建对象,如果"helloworld"这个字符串在java String池里不存在,会在java String池创建这个一个String对象("helloworld").如果已经存在,s1直接reference to 这个String池里的对象。
通过new赋值(String s2 = new String("helloworld")) 至少创建一个对象,也可能两个。因为用到new 关键字,会在heap创建一个 str2 的String 对象,它的value 是 "helloworld".同时,如果"helloworld"这个字符串在java String池里不存在,会在java String池创建这个一个String对象("helloworld").