创建一个String对象一般有以下两种方式:

String str1 = "abcd";  
String str2 = new String("abcd");



这两种方式有什么区别呢?我们可以通过下面两个小例子来说明.

Example 1:


String a = "abcd";  
String b = "abcd";  
System.out.println(a == b);  // True  
System.out.println(a.equals(b)); // True


这里 a == b 为 true,是因为 a 和 b 都引用了方法区的同一个字符串字面值。在java中,当同一个字符串字面值出现多次时,在内存中只会保存该字符串的一份拷贝,即 a 指向的 "abcd" 跟 b 指向的 "abcd" 在内存中是同一个对象。这种方法叫做 "字符串驻留(string interning)"。

Example 2:


String c = new String("abcd");  
String d = new String("abcd");  
System.out.println(c == d);  // False  
System.out.println(c.equals(d)); // True


我们知道,new运算符的作用是在堆内存中创建一个新的对象。因而这里两次 new 就会在堆中创建两个String对象,尽管它们的字面值是相同的,但其实是两个同不的对象,占两份空间。因此 c == d 自然会返回 false 了。图示如下:


Java中用引号创建String对象和用构造函数的区别_字符串

此外,我们可以调用String的intern()方法来强制让JVM使用字符串驻留,例如下面的代码:



String c = new String("abcd").intern();  
String d = new String("abcd").intern();  
System.out.println(c == d);  // Now true  
System.out.println(c.equals(d)); // True



intern()方法调用时,首先会在字符串常量池中检查有没有 "abcd".equals(常量池中的字符串)为true的对象,如果有,则把该对象返回,如果没有,则创建一个"abcd"对象放到常量池中。因此,在创建d 的时候,intern()就会把先前创建的 "abcd" 的引用直接返回了。