在Java中使用字符串有一個(gè)非常重要的規(guī)則必須記得,一個(gè)字符串對(duì)象一旦被配置,它的內(nèi)容就是固定不可變的。例如下面這個(gè)聲明:
String str = "caterpillar";
這個(gè)聲明會(huì)配置一個(gè)長(zhǎng)度為11、內(nèi)容為caterpillar的字符串對(duì)象,您無(wú)法改變這個(gè)字符串對(duì)象的內(nèi)容。不要以為下面的語(yǔ)句就是改變一個(gè)字符串對(duì)象的內(nèi)容:
String str = "Just";
str = "Justin";
事實(shí)上在這個(gè)程序片段中會(huì)有兩個(gè)字符串對(duì)象,一個(gè)是Just字符串對(duì)象,長(zhǎng)度為4;一個(gè)是Justin字符串對(duì)象,長(zhǎng)度為6,兩個(gè)是不同的字符串對(duì)象。您并不是在Just字符串后加上in字符串,而是讓str名稱引用自Justin字符串對(duì)象。圖6-2展示出了這個(gè)概念。

圖6-2 使用=作字符串指定的意義
在Java中,使用“=”將一個(gè)字符串對(duì)象指定給一個(gè)變量名稱,其意義為改變?cè)撁Q所引用的對(duì)象,原來(lái)被引用的字符串對(duì)象若沒(méi)有其他名稱來(lái)引用它,就會(huì)在適當(dāng)?shù)臅r(shí)候被Java的“垃圾回收”(Garbage Collection)機(jī)制回收。在Java中,程序設(shè)計(jì)人員通常不用關(guān)心無(wú)用對(duì)象的資源釋放問(wèn)題,Java會(huì)檢查對(duì)象是否不再被引用,沒(méi)有被任何名稱引用的對(duì)象將會(huì)被回收。
在Java執(zhí)行時(shí)會(huì)維護(hù)一個(gè)String池(Pool)。對(duì)于一些可以共享的字符串對(duì)象,會(huì)先在String池中查找是否存在相同的String內(nèi)容(字符相同),如果有就直接返回,而不是直接創(chuàng)造一個(gè)新的String對(duì)象,以減少內(nèi)存的耗用。如果在程序中使用下面的方式來(lái)聲明,則實(shí)際上是指向同一個(gè)字符串對(duì)象:
String str1 = "flyweight";
String str2 = "flyweight";
System.out.println(str1 == str2);
當(dāng)直接在程序中使用""來(lái)包括一個(gè)字符串時(shí),該字符串就會(huì)在String池中,上面的程序片段中的flyweight就是這樣的情況。flyweight在程序中會(huì)有一個(gè)實(shí)例,而str1與str2同時(shí)引用自flyweight。圖6-3表示了這個(gè)概念。

圖6-3 字符串變量str1、str2同時(shí)引用自flyweight
在Java中如果==被使用于兩個(gè)對(duì)象類型的變量,它是用于比較兩個(gè)變量是否引用自同一對(duì)象,所以在圖6-3中當(dāng)str1==str2比較時(shí),程序的執(zhí)行結(jié)果會(huì)顯示true。
再來(lái)看看關(guān)于String的intern()方法,先看一下API說(shuō)明的節(jié)選(翻譯):
在intern()方法被調(diào)用時(shí),如果池(Pool)中已經(jīng)包括了相同的String對(duì)象(相同與否由equals()方法決定),那么會(huì)從池中返回該字符串,否則原String對(duì)象會(huì)被加入池中,并返回這個(gè)String對(duì)象的引用。
這段話其實(shí)說(shuō)明了Flyweight模式(參見(jiàn)本章的網(wǎng)絡(luò)索引) 的運(yùn)作方式,直接使用范例6.4來(lái)說(shuō)明會(huì)更清楚。
ü 范例6.4 InternString.java
public class InternString {
public static void main(String[] args) {
String str1 = "fly";
String str2 = "weight";
String str3 = "flyweight";
String str4 = null;
str4 = str1 + str2;
System.out.println(str3 == str4);
str4 = (str1 + str2).intern();
System.out.println(str3 == str4);
}
}
style='font-family:宋體'>使用圖形來(lái)說(shuō)明這個(gè)范例,在聲明str1、str2、str3后,String池中的狀況如圖6-4所示。


圖6-4 字符串池示意
在Java中,使用+串聯(lián)字符串會(huì)產(chǎn)生一個(gè)新的字符串對(duì)象,所以在程序中第一次比較str3與str4對(duì)象是否為同一對(duì)象時(shí),結(jié)果會(huì)是false,如圖6-5所示。

圖6-5 字符串加法會(huì)產(chǎn)生新的字符串
intern()方法會(huì)先檢查String池中是否存在字符部分相同的字符串對(duì)象,如果有就返回。由于程序中之前已經(jīng)聲明flyweight字符串對(duì)象,intern()在String池中發(fā)現(xiàn)了它,所以直接返回。這時(shí)再進(jìn)行比較,str3與str4所指向的其實(shí)會(huì)是同一對(duì)象,所以結(jié)果會(huì)是true,如圖6-6所示。

圖6-6 intern()會(huì)返回String池中字符串對(duì)象的引用
由以上的說(shuō)明也得出一個(gè)結(jié)論,==在Java中被用來(lái)比較兩個(gè)變量名稱是否引用自同一對(duì)象,所以不可以用==來(lái)比較兩個(gè)字符串的字符內(nèi)容是否相同。例如:
String str1 = new String("caterpillar");
String str2 = new String("caterpillar");
System.out.println(str1 == str2);
雖然兩個(gè)字符串對(duì)象的字符值完全相同,但實(shí)際上在這個(gè)程序片段中,產(chǎn)生了兩個(gè)String的實(shí)例,str1與str2分別引用自不同的實(shí)例。所以使用==比較時(shí)結(jié)果會(huì)顯示false。如果要比較兩個(gè)字符串對(duì)象的字符值是否相同,要使用equals()方法,以下的寫(xiě)法才會(huì)顯示true的結(jié)果:
String str1 = new String("caterpillar");
String str2 = new String("caterpillar");
System.out.println(str1.equals(str2));