用以下几种情况的代码及分析大致讲解 String s = "abc" 和 String s = new String("abc") 的区别。
情况1:
(1)
String str1 = "abc";
System.out.println(str1 == "abc");
步骤:
- 栈中开辟一块空间存放引用str1;
- String池中开辟一块空间,存放String常量"abc";
- 引用str1指向池中String常量"abc";
- str1所指代的地址即常量"abc"所在地址,输出为true;
情况2:
(2)
String str2 = new String("abc");
System.out.println(str2 == "abc");
步骤:
- 栈中开辟一块空间存放引用str2;
- 堆中开辟一块空间存放一个新建的String对象"abc";
- 引用str2指向堆中的新建的String对象"abc";
- str2所指代的对象地址为堆中地址,而常量"abc"地址在池中,输出为false;
注意:对于通过 new 产生的对象,会先去常量池检查有没有 “abc”,如果没有,先在常量池创建一个 “abc” 对象,然后在堆中创建一个常量池中此 “abc” 对象的拷贝对象。
有道面试题: String s = new String(“xyz”); 产生几个对象?
一个或两个。如果常量池中原来没有 ”xyz”, 就是两个。如果原来的常量池中存在“xyz”时,就是一个。
对于基础类型的变量和常量:变量和引用存储在栈中,常量存储在常量池中。
情况3、4:
(3)
String str1 = "a";
String str2 = "b";
String str3 = str1 + "b";
//str1 和 str2 是字符串常量,所以在编译期就确定了。
//str3 中有个 str1 是引用,所以不会在编译期确定。
//又因为String是 final 类型的,所以在 str1 + "b" 的时候实际上是创建了一个新的对象,在把新对象的引用传给str3。
(4)
final String str1 = "a";
String str2 = "b";
String str3 = str1 + "b";
//这里和(3)的不同就是给 str1 加上了一个final,这样str1就变成了一个常量。
//这样 str3 就可以在编译期中就确定了
情况5:
intern()方法:当调用 intern 方法时,如果常量池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回常量池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。
(5)
String str1 = "ab";
String str2 = new String("ab");
System.out.println(str1== str2);//false
System.out.println(str2.intern() == str1);//true
System.out.println(str1== str2);//false
str2 = str2.intern();
System.out.println(str1== str2);//true
讲解:
str1 指向的是常量池对象 "ab";
str2 指向的是堆中的对象 "ab";
调用了 str2 = str2.intern() 后,str2.intern()判断常量池中是否有 "ab"对象,如果有就返回,没有就创建并返回,此时就返回的 str1 所指向的那个对象 "ab" 。
所以 str1 == str2;
参考文章
http://www.cnblogs.com/wanlipeng/archive/2010/10/21/1857513.html
http://blog.csdn.net/lubiaopan/article/details/4776000/
http://blog.csdn.net/u014082714/article/details/50087563