integer數據對比
對于Integer var = ? 在-128至127范圍內的賦值,Integer對象是在IntegerCache.cache產生,會復用已有對象,這個區間內的Integer值可以直接使用==進行判斷,但是這個區間之外的所有數據,都會在堆上產生,并不會復用已有對象,這是一個大坑,推薦使用equals方法進行判斷。
Array的subList
ArrayList的subList結果不可強轉成ArrayList,否則會拋出ClassCastException異常,即java.util.RandomAccessSubList cannot be cast to java.util.ArrayList. 說明:subList 返回的是 ArrayList 的內部類 SubList,并不是 ArrayList ,而是 ArrayList 的一個視圖,對于SubList子列表的所有操作最終會反映到原列表上。
List變Array
使用toArray帶參方法,入參分配的數組空間不夠大時,toArray方法內部將重新分配內存空間,并返回新數組地址;如果數組元素個數大于實際所需,下標為[ list.size() ]的數組元素將被置為null,其它數組元素保持原值,因此最好將方法入參數組大小定義與集合元素個數一致。直接使用toArray無參方法存在問題,此方法返回值只能是Object[]類,若強轉其它類型數組將出現ClassCastException錯誤。
正例:
1 List<String> list = new ArrayList<String>(2);
2 list.add("guan");
3 list.add("bao");
4 String[] array = new String[list.size()];
5 array = list.toArray(array);
Array變List
使用工具類Arrays.asList()把數組轉換成集合時,不能使用其修改集合相關的方法,它的add/remove/clear方法會拋出UnsupportedOperationException異常。 說明:asList的返回對象是一個Arrays內部類,并沒有實現集合的修改方法。Arrays.asList體現的是適配器模式,只是轉換接口,后臺的數據仍是數組。
1 String[] str = new String[] { "you", "wu" };
2 List list = Arrays.asList(str);
第一種情況:list.add("yangguanbao"); 運行時異常。
第二種情況:str[0] = "gujin"; 那么list.get(0)也會隨之修改。
泛型
泛型通配符<? extends T>來接收返回的數據,此寫法的泛型集合不能使用add方法,而<? super T>不能使用get方法,作為接口調用賦值時易出錯。 說明:擴展說一下PECS(Producer Extends Consumer Super)原則:第一、頻繁往外讀取內容的,適合用<? extends T>。第二、經常往里插入的,適合用<? super T>。
Foreach
不要在foreach循環里進行元素的remove/add操作。remove元素請使用Iterator方式,如果并發操作,需要對Iterator對象加鎖。
正例:
1 Iterator<String> iterator = list.iterator();
2 while (iterator.hasNext()) {
3 String item = iterator.next();
4 if (刪除元素的條件) {
5 iterator.remove();
6 }
7 }
反例:
1 List<String> list = new ArrayList<String>();
2 list.add("1");
3 list.add("2");
4 for (String item : list) {
5 if ("1".equals(item)) {
6 list.remove(item);
7 }
8 }
排序
在JDK7版本及以上,Comparator要考慮相等情況,不然Arrays.sort,Collections.sort會報IllegalArgumentException異常,下例中沒有處理相等的情況,實際使用中可能會出現異常
1 new Comparator<Student>() {
2 @Override
3 public int compare(Student o1, Student o2) {
4 return o1.getId() > o2.getId() ? 1 : -1;
5 }
6 };
遍歷Map
使用entrySet遍歷Map類集合KV,而不是keySet方式進行遍歷。keySet其實是遍歷了2次,一次是轉為Iterator對象,另一次是從hashMap中取出key所對應的value。而entrySet只是遍歷了一次就把key和value都放到了entry中,效率更高。如果是JDK8,使用Map.foreach方法。