Java常见陷阱
總覽
Java是一種極簡主義的語言,具有比其他語言故意更少的功能,盡管如此,Java仍然具有產生奇怪效果的邊緣情況,甚至具有令人驚訝的效果的一些常見情況也會使您輕而易舉。 如果您習慣于閱讀另一種語言,則可以輕松地以錯誤的方式閱讀Java,而不會感到困惑。
變量僅是引用或原語
沒錯,變量不是對象。 這意味著當您看到以下內容時,s 不是一個對象 ,它不是String,它是對String的引用
String s = "Hello";這回答了許多混亂的領域,例如;
- 問:如果String是不可變的,如何更改它。 例如 s + =“!”;
- 答:在普通的Java中,您只能更改對String的引用。
==比較引用,而不是它們的內容。
更令人困惑的是,有時使用==是可行的。 如果您有兩個相同的不可變值,那么JVM也可以嘗試使引用相同。 例如
String s1 = "Hi", s2 = "Hi";Integer a = 12, b = 12;在這兩種情況下,都使用對象池,因此引用最終是相同的。 s1 == s2和a == b都成立,因為JVM引用了同一對象。 但是,稍微改變一下代碼,以便JVM不會合并對象,并且==返回false,這可能是意外的。 在這種情況下,您需要使用equals。
String s3 = new String(s1);Integer c = -222, d = -222;s1 == s2 // is trues1 == s3 // is falses1.equals(s3) // is truea == b // is truec == d // is false (different objects were created)c.equals(d) // is true對于Integer,對象池從-128開始,至少為127(可能更高)
Java按值傳遞引用
所有變量都按值傳遞,甚至引用也是如此。 這意味著當您有一個變量作為對對象的引用時,將復制此引用,但不復制該對象。 例如
public static void addAWord(StringBuilder sb) {sb.append(" word");sb = null; }StringBuilder sb = new StringBuilder("first "); addWord(sb); addWord(sb); System.out.println(sb); // prints "first word word"引用的對象可以更改,但是對復制的引用的更改對調用者無效。
在大多數JVM中,Object.hashCode()與內存位置無關
hashCode()必須保持不變。 沒有這個事實,像HashSet或ConcurrentHashMap這樣的哈希集合將無法工作。 但是,對象可以位于內存中的任何位置,并且可以更改位置,而無需您的程序知道發生了這種情況。 使用該位置的hashCode無效(除非您有一個不移動對象的JVM)
對于OpenJDK和HotSpot JVM,hashCode()是按需生成的,并存儲在對象的標頭中。 使用Unsafe,您可以查看是否已設置hashCode(),甚至可以通過
Object.toString()做一些令人驚訝的事,而不是有用的事
toString()的默認行為是為類和hashCode()打印內部名稱。
如前所述,hashCode不是存儲位置,即使它以十六進制打印也是如此。 同樣,類名,特別是對于數組,也令人困惑。 例如; String []打印為[Ljava.lang.String; [表示它是一個數組,L表示它是一個“語言”創建的類,而不是像BTW這樣具有代碼B的字節之類的基元。 表示課程結束。 例如說你有一個像
String[] words = { "Hello", "World" }; System.out.println(words);打印類似
[Ljava.lang.String;@45ee12a7不幸的是,您必須知道該類是一個對象數組,例如,如果您只有對象字,那么您將遇到問題,并且必須知道調用Arrays.toString(words)。 這種中斷封裝的方式非常糟糕,并且在StackOverflow上經常造成混亂。
我曾問過甲骨文公司的其他開發人員有關此問題,我的印象是現在很難修復它。
翻譯自: https://www.javacodegeeks.com/2014/03/common-gotchas-in-java.html
總結