很多老問題一直被新手提出來問 我把我之前在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)
沒有留言:
張貼留言