淺談Java包裝類型Long的==操作引發的低級bug
背景
一個簡單的列表檢索功能,列表元素有一個 Long 類型的屬性,遍歷過程中犯瞭一個低級錯誤,導致功能流程始終錯誤,本文將分享兩個低級錯誤引發的 bug。
兩個 Long 類型的 ==
查找某個元素 A 在列表 B 中對應的對象的時候,根據元素主鍵查詢,主鍵類型為包裝類型 Long ,遍歷流程如下:
for(MyData temp:b){ if (temp.getId() == a.getId() { // MyData 的 id 屬性為 Long 類型 return temp; } }
這麼一段簡單的查找代碼,結果怎麼都找不到目標對象,斷點調試發現問題出在 == 操作上,改成 equals 就可以瞭。
關於 Java Long 的包裝類型和元素類型的判斷相等的操作回顧:
Long a = 81487354807713792L; Long b = 81487354807713792L; System.out.println(a==b); // false System.out.println(a.equals(b)); // true long c = 81487354807713792L; long d = 81487354807713792L; System.out.println(c==d); // true
對 Collections.EMPTY_SET 進行 add 引發的異常
另一個低級錯誤是對 Collections.EMPTY_SET 進行 add 引發的,需要合並兩個集合,第一個集合 A 可能是
Collections.EMPTY_SET ,最終將另一個集合 B 合並到 A 得到一個大集合。 Set<MyData> a = getDatas();// 如果為空,返回瞭 Collections.EMPTY_SET Set<MyData> b = getDatas1(); a.addAll(b);
當集合 a 為集合的空對象時,操作異常:
Exception in thread “main” java.lang.UnsupportedOperationException
at java.util.AbstractCollection.add(AbstractCollection.java:262)
修正方式:如果需要直接對一個集合進行 add ,就不能用 Collections.EMPTY_SET。
Collections 的空集合使用註意事項
以 Collections.EMPTY_SET 為例,跟源碼它的定義主要是下面三行代碼:
public static final Set EMPTY_SET = new EmptySet<>(); private static class EmptySet<E> extends AbstractSet<E> AbstractSet<E> extends AbstractCollection
核心在於 AbstractCollection 類的 add,默認直接拋出瞭異常,限制瞭空集合不允許添加:
public boolean add(E e) { throw new UnsupportedOperationException(); }
結論:java.util.Collections 類中所有的 EMPTY_XXX 對象都不能進行 add 操作。
啟示錄
定位到這兩個低級錯誤後,想起那句調侃:代碼編寫分分鐘,bug 查找兩小時。這兩個問題恰好是一個比較復雜的流程的一部分,構建環境進行測試,測一次差不多十幾分鐘,加上機器怠工,跟這倆小問題,耗瞭兩個小時。
到此這篇關於淺談Java包裝類型Long的==操作引發的低級bug的文章就介紹到這瞭,更多相關Java Long的==內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Java Collections的emptyList、EMPTY_LIST詳解與使用說明
- Java日常練習題,每天進步一點點(14)
- Java中Collections.emptyList()的註意事項
- Java日常練習題,每天進步一點點(54)
- Java 數組轉List的四種方式小結