问题:String str1 = “abc”;与
String str1 = new String(“abc”);有什么区别?
解答问题之前来一些准备知识。
先明确两句话:(记住这两句话,后面会用到)
1.字符串常量存储在字符常量池,目的是共享
2.字符串非常量对象存储在堆中
String的实例化方式:
方式一:通过字面量定义的方式。
方式二:通过new+构造器的方式。
我们先看一个例子:
String s1 = "javaEE";
String s2 = "javaEE";
String s3 = new String("javaEE");
String s4 = new String("javaEE");
System.out.println(s1==s2);//true
System.out.println(s1==s3);//false
System.out.println(s1==s4);//false
System.out.println(s3==s4);//false
运行结果:
true
false
false
false
为什么会出现这样的结果呢?
因为:用方式二:new+构造器的方式定义的String是非常量,而用方式一:字面量赋值的方式定义的String是常量。而根据文章开头的两句话,非常量对象是存储在堆中的,换句话说,用new+构造器方式定义的String都是存储在堆中的,所以你用堆中存储的地址去和常量池中存储的地址作比较,结果当然是false。这也解释了第二个和第三个为什么是false。至于第四个,堆中没有规定不能有相同的值,所以会在堆中分别创建两个不同的地址储存两个值,故第四个也是false。
关于第一个为什么是true,是因为:使用字面量的方式(方式一)给字符串赋值,此时字符串是存储在常量池中的,而常量池中只能有相同的一个值,如果再有相同内容的String,常量池会把池中已有的常量地址赋给需要赋值的String。。所以s1,s2储存的地址都是常量池中的同一个地址,所以第一个结果是true。
再来个一迷惑性的问题:
Person p1 = new Person("Tom",12);
Person p2 = new Person("Tom",12);
System.out.println(p1.name==p2.name);
运行结果:
true
这时,p1,p2在堆中存储的地址都是Tom在常量池中的同一个地址,所以结果是true。
所以,开头的问题显然答案很明显,字面量赋值创建的String存储在常量池中,且常量池中内容不能重复。new+构造器创建的String存储在堆中。
另外几个需要知道的点:
1.常量与常量拼接的结果在常量池,且常量池中不会存在相同内容。
2.只要拼接的内容有一个是变量,结果就在堆中。
3.如果拼接的结果调用intern()方法,返回值就在常量池中。