创建一个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 了。图示如下:
此外,我们可以调用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" 的引用直接返回了。