Java開發者犯的錯誤Top10

2019-08-12     進擊的IT程式設計師

Top1. 數組轉換為數組列表

將數組轉換為數組列表,開發者經常會這樣做:

[java]List list = Arrays.asList(arr);

Arrays.asList()將返回一個數組內部是私有靜態類的ArrayList,這不是java.util.ArrayList類,java.util.Arrays.ArrayList類有set()、 get()、 contains()方法,但是沒有任何加元素的方法,因此它的大小是固定的。可以用於方便轉換list,不能添加新元素,所以選擇正確的使用場景。

你也可以這麼做來創建一個真正的數組:

[java]ArrayList arrayList = new ArrayList(Arrays.asList(arr));

ArrayList的構造函數能夠接受一個集合類型,這也是java.util.Arrays.ArrayList的超級類型。


Top2. 檢查一個數組包含一個值

開發者經常這麼做:

[java]Set set = new HashSet(Arrays.asList(arr));return set.contains(targetValue);

代碼可以工作,但是沒有必要首先轉換列表到Set,轉換一個列表到一個Set需要額外的時間。因此你可以把它簡化為:

[java]Arrays.asList(arr).contains(targetValue);

[java]for(String s: arr){if(s.equals(targetValue))return true;}return false;

第一個比第二個更具可讀性


Top3. 在一個循環中從一個列表里刪除一個元素

考慮下面刪除元素的代碼在疊代中的結果:

[java]ArrayList list = new ArrayList(Arrays.asList("a", "b", "c", "d"));for (int i = 0; i < list.size(); i++) {list.remove(i);}System.out.println(list);

輸出是:

[java][b, d]

該方法有一個嚴重的問題,當一個元素被刪除時,列表收縮的大小以及指針改變了。所以想要在循環內利用指針刪除多個元素是無法正常進行的。

這種情況下使用疊代器才是正確的方法,foreach循環在Java中的工作像是一個疊代器,但實際上並不是,考慮下面的代碼:

[java]ArrayList list = new ArrayList(Arrays.asList("a", "b", "c", "d"));for (String s : list) {if (s.equals("a"))list.remove(s);}

它會報出ConcurrentModificationException異常。

相反下面這個就可以正常工作。

[java]ArrayList list = new ArrayList(Arrays.asList("a", "b", "c", "d"));Iterator iter = list.iterator();while (iter.hasNext()) {String s = iter.next();if (s.equals("a")) {iter.remove();}}

.next()必須在.remove()之前被調用。在foreach循環中,編譯器將在刪除元素操作之後調用.next(),這也是導致ConcurrentModificationException異常的原因,你可以點擊此處查看ArrayList.iterator()的原始碼。


Top4. Hashtable vs HashMap

根據算法的常規,Hashtable是對數據結構的稱呼。但是在Java中,數據結構的名稱是HashMap。Hashtable和HashMap關鍵不同之一是Hashtable是同步的。


Top5. 使用集合的原始類型

在Java中,原始類型和無限制的通配符類型很容易被混淆。以Set為例,Set是原始類型,而Set(?)則是無限制的通配符類型。

考慮下面的代碼,以一個原始類型List作為參數:

[java]public static void add(List list, Object o){list.add(o);}public static void main(String[] args){List list = new ArrayList();add(list, 10);String s = list.get(0);}

該代碼會拋出一個異常:

[java]Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Stringat ...

使用原始類型集合是危險的,因為原始類型集合跳過了泛型類型檢查,也不安全。Set、Set和Set之間有很大的不同。


Top6. 訪問級別

開發者經常對類域使用public,這很容易通過直接引用獲得域值,但這是一個非常糟糕的設計。根據經驗來說是給予成員的訪問級別越低越好。

詳細情況可點擊查看Java中成員訪問級別(請點擊「閱讀原文」查看):public、protected、private


Top7.ArrayList VS LinkedList

如果你不知道ArrayList和LinkedList之間的區別時,你可能會經常的選用ArrayList,因為它看起來看熟悉。然而它們之間有巨大的性能不同。簡單的來說,如果有大量的添加/刪除操作,並且沒有很多的隨機存取操作時,LinkedList應該是你的首選。如果您對此不是很了解的話,點此此處查看更多關於它們性能的信息。


Top8. Mutable VS Immutable

Immutable對象有很多優勢,比如簡單、安全等等。但它要求每一個不同的值都需要有一個不同的對象,而太多的對象可能會導致垃圾收集的高成本。所以對Mutable和Immutable的選擇應該有一個平衡點。

一般來說,Mutable對象用於避免產生過多的中間對象,經典的例子是連接大量的字符串數。如果你使用Immutable字符串,那麼會產生很多符合垃圾收集條件的對象。這對CPU是浪費時間和精力的,當其可以使用Mutable對象作為正確的解決方案。(如StringBuilder)

[java]String result="";for(String s: arr){result = result + s;}

這裡還有一些其他Mutable對象可取的情況。例如mutable對象傳遞到方法中允許你在不跳過太多語法的情況下收集多個結果。另一個例子是排序和過濾,你可以構建一個帶有原有集合的方法,並返回一個已排序的,不過這對大的集合來說會造成更大的浪費。


Top9. Super和Sub構造函數

這個編譯錯誤是因為默認的Super構造函數是未定義的。在Java中,如果一個類沒有定義一個構造函數,編譯器會默認的為類插入一個無參數構造函數。如果一個構造函數是在Super類中定義的,這種情況下Super(String s),編譯器不會插入默認的無參數構造函數。

另一方面,Sub類的構造函數,無論帶不帶有參數,都會調用無參數的Super構造函數。

編譯器在Sub類中試圖將Super()插入到兩個構造函數中,但是Super默認的構造函數是沒有定義的,編譯器才會報錯。如何解決這一問題?你只需在Super類中添加一個Super()構造函數,如下所示:

[java]public Super(){System.out.println("Super");}

或移除自定義的Super構造函數,又或者在Sub函數中添加super(value)。


Top10. ""或構造函數?

字符串可以通過兩種方式創建:

[java]//1. use double quotesString x = "abc";//2. use constructorString y = new String("abc");

它們之間有何不同?下面的例子可以給出答案:

[java]String a = "abcd";String b = "abcd";System.out.println(a == b); // TrueSystem.out.println(a.equals(b)); // TrueString c = new String("abcd");String d = new String("abcd");System.out.println(c == d); // FalseSystem.out.println(c.equals(d)); // True

私信我:「資料」,可免費領取更多學習資料哦

文章來源: https://twgreatdaily.com/UcIr6W0BMH2_cNUgMvFW.html