很多老問題一直被新手提出來問
我把我之前在Java週報上發表過舊的文章貼上來
希望對初學者觀念的釐清有點幫助
眨在這,下次再問類似的問題
browser就別客氣,殺無赦吧~~~~
-------------------------------
原文刊登在 2002/09/12 Java 週報上
初學Java的人﹐在學習的過程中遇到的第一個不能理解的問題﹐通常就是
字串的比較問題﹐因為他們實在不能理解明明兩個"一樣"的字串﹐為什麼
有時在做相等的比較運算時會得到ture的結果﹐可是有時得到的結果確是
false?這個問題在網路上常常都被提出來討論﹐已經算是一個FAQ到不能
FAQ的問題﹐也許是那些初學者們都沒有買到一本好的教科書﹐書中並沒有
教導他們正確的認識字串﹐要是有認真地讀完筆者寫的書﹐應該就不會被
這個問題所困擾了。好吧!大家都是窮學生買不起五百塊以上的書﹐那筆
者就藉著Java週報的一角﹐詳細地說明字串的比較運算。
嚴格說起來﹐字串在Java中也是一個完整的物件﹐兩個物件要做比較時﹐
首先你得先想清楚你想要比較的東西是什麼。舉個例子來說吧﹐日常生活
中婆婆媽媽們聚在一起總是喜歡比較來比較去的﹐有時後比的是誰家的孩
子功課比較好啦﹐誰的老公賺的錢比較多啦等…。人也是物件的一種﹐當
你把兩個人做比較時﹐你一定是把這兩個人身上的某一個屬性拿出來比較
﹐比如說身高、體重、成績等等﹐所以你要說這兩個人完全相同時﹐一定
是你所拿出來比較的每個屬性的內容都相同。另外一種觀點的比較方式是
﹐一個人有時會有好幾個不同的綽號﹐別的人要經由比較確認之後﹐才知
道原來這幾個綽號指的都是你這個人。
話說回來﹐在Java程式中﹐你要比較兩個物件時﹐首先你要確認要比較的
是兩件物件的內容?還是比較兩個物件變數是否指向同一個物件?如果你
是要比較兩個物件的內容時﹐請使用該物件所提供的"equals"方法。例如:
String str1 = new String("hello");
String str2 = new String("hello");
boolean b1 = str1.equals(str2);
booealn b2 = str2.equals(str1);
上例中用誰的equals方法來使用﹐結果都是一樣的(都是true)﹐不過如果
str1和str2是兩個不同的類別物件﹐意義可就不太一樣了﹐這個問題有空
再說。如果你比較個物件變數指的是不是同一個物件﹐那你就得使用比較
算的"=="運算子。例如:
String str3 = new String("hello");
String str4 = new String("hello");
String str5 = str3;
boolean b3 = str3 == str4;
boolean b4 = str3 == str5;
b3的結果是false﹐b4的結果是true。不知道筆者這樣的解說﹐大家應該
可以清楚的知道什麼時候要使用equals方法﹐又什麼時後要使用"=="運算子。
話再說回來﹐會令初學者感到困惑的是類似下面的例子:
String str6 = "hello";
String str7 = "hello";
boolean b5 = str6.equals(str7);
boolean b6 = str6 == str7;
b5的結果是true﹐b6的結果也是true。為什麼b3的結果會是false﹐而b6
的結果卻是true呢?上面幾個例子中的字串物件﹐不都是"hello"這個字
串嗎?你只說對了一半﹐上面例子中str1到str7所指向的字串物件﹐它
們的"內容"都是"hello"這幾個字母沒錯﹐可是它們指向的可不都是同一
個字串物件!str1到str4這四個物件變數指向的字串物件﹐是經由標準
的物件產生方式(new)所產生出來的﹐就算內容一樣﹐可是還是不同的物
件。就像兩個雙胞胎來說﹐甚至兩個用複製出來的人﹐就算所有的屬性
都一樣﹐兩個人還是不同的個體﹐在地球上佔用不同的空間。==運算子
既然比較的是兩個物件變數是否指向的是同一個物件﹐所以b3的結果會
是false就是這個原因囉。
那b6的結果為什麼會是true呢?這是Java為了效能的考量﹐而使用的一
種技巧。因為字串對於一個應用程式來說﹐使用上是非常的頻繁的﹐如
果每個相同內容的字串物件﹐都佔用不同的記憶空間不是很浪費嗎?而
且JVM也要去處理這些可能只用過一次就不再使用的字串。因此﹐當你用
雙引號來產生一個字串物件時﹐JVM會先到記憶體中一個名叫String Pool
的地方去查詢一下﹐是否裡面已經有這個字串了﹐如果有則直接拿出來
使用;如果沒有就產生一個新的放到裡面去。所以str6和str7指向的字
串物件﹐都是在Stirng Pool中的hello字串物件﹐既然兩個物件變數指
向的是同一個物件﹐那麼用==運算子比較的結果當然就是true了。
被搞的昏頭轉向了嗎?其實就記住筆者最前面所說的﹐當你要比較的是
兩個物件的內容相不相等時﹐請用equals方法﹐如果要比較的是兩個物
件變數指向的是不是同一個物件時、請用==運算子。這樣一來﹐不管是
一般的物件還是字串物件在比較時﹐你就不會感到困惑了。
piggy edited on 2003-09-21 00:17
An Apple a day, keeps M$ away
作者 Re:[新手必看] Java字串的比較 [Re:piggy]
getch
發文: 15
積分: 0
於 2004-05-04 17:06 收藏文章? |
但這種情形
String str1=new String("hello");
String str2="hello";
boolean arg=(str1==str2);
又會得到false
是因為new所產生的物件不是在String pool的關係,所以str2在String pool找不到相同的字串,因此才創造一個新的物件嗎?
作者 Re:[新手必看] Java字串的比較 [Re:piggy]
hellfire
大爆炸!!??
發文: 24
積分: 0
於 2004-05-04 21:17 收藏文章? |
str1做了兩件事
創造"hello"字串放到pool中
再根據它的內容做出一個新的字串
str2則是從pool中取得"hello"
所以才會得到false
作者 Re:[新手必看] Java字串的比較 [Re:hellfire]
hkdennis2k
發文: 1069
積分: 4
於 2004-11-08 12:33 收藏文章? |
hellfire wrote:
str1做了兩件事
創造"hello"字串放到pool中
再根據它的內容做出一個新的字串
str2則是從pool中取得"hello"
不,
在compile 時, java 檔內的所有 hardcode 了的 string
會被找出來, 作為 const, 共用一個 char[]
比較像這個 1
2
3
4 static final String _STRING1=new String("hello".toCharArray());
.......
String str1=new String(_STRING1);
String str2=_STRING1;
作者 Re:[新手必看] Java字串的比較 [Re:hkdennis2k]
Duncan
等待是一生最初的蒼老
版主
發文: 4482
積分: 38
於 2004-11-12 03:05 收藏文章? |
hkdennis2k wrote:
不,
在compile 時, java 檔內的所有 hardcode 了的 string
會被找出來, 作為 const, 共用一個 char[]
比較像這個 1
2
3
4 static final String _STRING1=new String("hello".toCharArray());
.......
String str1=new String(_STRING1);
String str2=_STRING1;
現在是在討論兩個 statement runtime 的行為,你講的是編譯期的事。
主要的差別在於使用 new operator 建構 String object 即使其內容在 VM runtime maintain 的 string pool 裡有相同的,依然會建構新的 String object,直接用 string literal 的寫法則是 VM 到 class const pool 去找對應的字串,如果 string pool 已經有相同內容的就使用 string pool 裡的,否則就依照 const pool 裡 string 內容建構 String object,並放到 string pool 裡。
2013年2月20日 星期三
Java字串的比較 [精
Labels:
Java
在文字的世界裡尋找心靈的共鳴,遠山藍以溫柔的筆觸分享書籍的力量與生活的智慧。無論是細膩的書評、深刻的人生感悟,還是技術與創新的新奇發現,每篇文章都是一次內心的療癒旅程。希望透過閱讀,帶領讀者在忙碌的生活中找到一片寧靜與啟發。讓我們一起,在書香中遇見更好的自己!
訂閱:
張貼留言 (Atom)
沒有留言:
張貼留言