成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 集合使用不當(dāng),Code Review 被 Diss了!

開發(fā) 后端
有很多小伙伴在寫代碼的時(shí)候,有一些比較基礎(chǔ)的問題沒有考慮到,導(dǎo)致項(xiàng)目 Code Review 的時(shí)候被 diss。

[[419682]]

大家好,我是 Guide!

有很多小伙伴在寫代碼的時(shí)候,有一些比較基礎(chǔ)的問題沒有考慮到,導(dǎo)致項(xiàng)目 Code Review 的時(shí)候被 diss。

上周五 Code Review 的時(shí)候,團(tuán)隊(duì)有個(gè)工作1年多的小伙伴使用 Java 集合的時(shí)候就出現(xiàn)了一個(gè)非常基礎(chǔ)的問題。

這篇文章我根據(jù)《阿里巴巴 Java 開發(fā)手冊(cè)》總結(jié)了關(guān)于集合使用常見的注意事項(xiàng)以及其具體原理。

強(qiáng)烈建議小伙伴們多多閱讀幾遍,避免自己寫代碼的時(shí)候出現(xiàn)這些低級(jí)的問題。

集合判空

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

判斷所有集合內(nèi)部的元素是否為空,使用 isEmpty() 方法,而不是 size()==0 的方式。

這是因?yàn)?isEmpty() 方法的可讀性更好,并且時(shí)間復(fù)雜度為 O(1)。

絕大部分我們使用的集合的 size() 方法的時(shí)間復(fù)雜度也是 O(1),不過,也有很多復(fù)雜度不是 O(1) 的,比如 java.util.concurrent 包下的某些集合(ConcurrentLinkedQueue 、ConcurrentHashMap...)。

下面是 ConcurrentHashMap 的 size() 方法和 isEmpty() 方法的源碼。

  1. public int size() { 
  2.     long n = sumCount(); 
  3.     return ((n < 0L) ? 0 : 
  4.             (n > (long)Integer.MAX_VALUE) ? Integer.MAX_VALUE : 
  5.             (int)n); 
  6. final long sumCount() { 
  7.     CounterCell[] as = counterCells; CounterCell a; 
  8.     long sum = baseCount; 
  9.     if (as != null) { 
  10.         for (int i = 0; i < as.length; ++i) { 
  11.             if ((a = as[i]) != null
  12.                 sum += a.value; 
  13.         } 
  14.     } 
  15.     return sum
  16. public boolean isEmpty() { 
  17.     return sumCount() <= 0L; // ignore transient negative values 

集合轉(zhuǎn) Map

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

在使用 java.util.stream.Collectors 類的 toMap() 方法轉(zhuǎn)為 Map 集合時(shí),一定要注意當(dāng) value 為 null 時(shí)會(huì)拋 NPE 異常。

  1. class Person { 
  2.     private String name
  3.     private String phoneNumber; 
  4.      // getters and setters 
  5.  
  6. List<Person> bookList = new ArrayList<>(); 
  7. bookList.add(new Person("jack","18163138123")); 
  8. bookList.add(new Person("martin",null)); 
  9. // 空指針異常 
  10. bookList.stream().collect(Collectors.toMap(Person::getName, Person::getPhoneNumber)); 

下面我們來解釋一下原因。

首先,我們來看 java.util.stream.Collectors 類的 toMap() 方法 ,可以看到其內(nèi)部調(diào)用了 Map 接口的 merge() 方法。

  1. public static <T, K, U, M extends Map<K, U>> 
  2. Collector<T, ?, M> toMap(Function<? super T, ? extends K> keyMapper, 
  3.                             Function<? super T, ? extends U> valueMapper, 
  4.                             BinaryOperator<U> mergeFunction, 
  5.                             Supplier<M> mapSupplier) { 
  6.     BiConsumer<M, T> accumulator 
  7.             = (map, element) -> map.merge(keyMapper.apply(element), 
  8.                                           valueMapper.apply(element), mergeFunction); 
  9.     return new CollectorImpl<>(mapSupplier, accumulator, mapMerger(mergeFunction), CH_ID); 

Map 接口的 merge() 方法如下,這個(gè)方法是接口中的默認(rèn)實(shí)現(xiàn)。

如果你還不了解 Java 8 新特性的話,請(qǐng)看這篇文章:《Java8 新特性總結(jié)》 。

  1. default V merge(K key, V value, 
  2.         BiFunction<? super V, ? super V, ? extends V> remappingFunction) { 
  3.     Objects.requireNonNull(remappingFunction); 
  4.     Objects.requireNonNull(value); 
  5.     V oldValue = get(key); 
  6.     V newValue = (oldValue == null) ? value : 
  7.                remappingFunction.apply(oldValue, value); 
  8.     if(newValue == null) { 
  9.         remove(key); 
  10.     } else { 
  11.         put(key, newValue); 
  12.     } 
  13.     return newValue; 

merge() 方法會(huì)先調(diào)用 Objects.requireNonNull() 方法判斷 value 是否為空。

  1. public static <T> T requireNonNull(T obj) { 
  2.     if (obj == null
  3.         throw new NullPointerException(); 
  4.     return obj; 

集合遍歷

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

不要在 foreach 循環(huán)里進(jìn)行元素的 remove/add 操作。remove 元素請(qǐng)使用 Iterator 方式,如果并發(fā)操作,需要對(duì) Iterator 對(duì)象加鎖。

通過反編譯你會(huì)發(fā)現(xiàn) foreach 語法糖底層其實(shí)還是依賴 Iterator 。不過, remove/add 操作直接調(diào)用的是集合自己的方法,而不是 Iterator 的 remove/add方法

這就導(dǎo)致 Iterator 莫名其妙地發(fā)現(xiàn)自己有元素被 remove/add ,然后,它就會(huì)拋出一個(gè) ConcurrentModificationException 來提示用戶發(fā)生了并發(fā)修改異常。這就是單線程狀態(tài)下產(chǎn)生的 fail-fast 機(jī)制。

fail-fast 機(jī)制 :多個(gè)線程對(duì) fail-fast 集合進(jìn)行修改的時(shí)候,可能會(huì)拋出ConcurrentModificationException。即使是單線程下也有可能會(huì)出現(xiàn)這種情況,上面已經(jīng)提到過。

Java8 開始,可以使用 Collection#removeIf()方法刪除滿足特定條件的元素,如

  1. List<Integer> list = new ArrayList<>(); 
  2. for (int i = 1; i <= 10; ++i) { 
  3.     list.add(i); 
  4. list.removeIf(filter -> filter % 2 == 0); /* 刪除list中的所有偶數(shù) */ 
  5. System.out.println(list); /* [1, 3, 5, 7, 9] */ 

除了上面介紹的直接使用 Iterator 進(jìn)行遍歷操作之外,你還可以:

  • 使用普通的 for 循環(huán)
  • 使用 fail-safe 的集合類。java.util包下面的所有的集合類都是 fail-fast 的,而java.util.concurrent包下面的所有的類都是 fail-safe 的。
  • ......

集合去重

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

可以利用 Set 元素唯一的特性,可以快速對(duì)一個(gè)集合進(jìn)行去重操作,避免使用 List 的 contains() 進(jìn)行遍歷去重或者判斷包含操作。

這里我們以 HashSet 和 ArrayList 為例說明。

  1. // Set 去重代碼示例 
  2. public static <T> Set<T> removeDuplicateBySet(List<T> data) { 
  3.  
  4.     if (CollectionUtils.isEmpty(data)) { 
  5.         return new HashSet<>(); 
  6.     } 
  7.     return new HashSet<>(data); 
  8.  
  9. // List 去重代碼示例 
  10. public static <T> List<T> removeDuplicateByList(List<T> data) { 
  11.  
  12.     if (CollectionUtils.isEmpty(data)) { 
  13.         return new ArrayList<>(); 
  14.  
  15.     } 
  16.     List<T> result = new ArrayList<>(data.size()); 
  17.     for (T current : data) { 
  18.         if (!result.contains(current)) { 
  19.             result.add(current); 
  20.         } 
  21.     } 
  22.     return result; 

兩者的核心差別在于 contains() 方法的實(shí)現(xiàn)。

HashSet 的 contains() 方法底部依賴的 HashMap 的 containsKey() 方法,時(shí)間復(fù)雜度接近于 O(1)(沒有出現(xiàn)哈希沖突的時(shí)候?yàn)?O(1))。

  1. private transient HashMap<E,Object> map; 
  2. public boolean contains(Object o) { 
  3.     return map.containsKey(o); 

我們有 N 個(gè)元素插入進(jìn) Set 中,那時(shí)間復(fù)雜度就接近是 O (n)。

ArrayList 的 contains() 方法是通過遍歷所有元素的方法來做的,時(shí)間復(fù)雜度接近是 O(n)。

  1. public boolean contains(Object o) { 
  2.     return indexOf(o) >= 0; 
  3. public int indexOf(Object o) { 
  4.     if (o == null) { 
  5.         for (int i = 0; i < size; i++) 
  6.             if (elementData[i]==null
  7.                 return i; 
  8.     } else { 
  9.         for (int i = 0; i < size; i++) 
  10.             if (o.equals(elementData[i])) 
  11.                 return i; 
  12.     } 
  13.     return -1; 

我們的 List 有 N 個(gè)元素,那時(shí)間復(fù)雜度就接近是 O (n^2)。

集合轉(zhuǎn)數(shù)組

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

使用集合轉(zhuǎn)數(shù)組的方法,必須使用集合的 toArray(T[] array),傳入的是類型完全一致、長(zhǎng)度為 0 的空數(shù)組。

toArray(T[] array) 方法的參數(shù)是一個(gè)泛型數(shù)組,如果 toArray 方法中沒有傳遞任何參數(shù)的話返回的是 Object類 型數(shù)組。

  1. String [] s= new String[]{ 
  2.     "dog""lazy""a""over""jumps""fox""brown""quick""A" 
  3. }; 
  4. List<String> list = Arrays.asList(s); 
  5. Collections.reverse(list); 
  6. //沒有指定類型的話會(huì)報(bào)錯(cuò) 
  7. s=list.toArray(new String[0]); 

由于 JVM 優(yōu)化,new String[0]作為Collection.toArray()方法的參數(shù)現(xiàn)在使用更好,new String[0]就是起一個(gè)模板的作用,指定了返回?cái)?shù)組的類型,0 是為了節(jié)省空間,因?yàn)樗皇菫榱苏f明返回的類型。詳見:https://shipilev.net/blog/2016/arrays-wisdom-ancients/

數(shù)組轉(zhuǎn)集合

《阿里巴巴 Java 開發(fā)手冊(cè)》的描述如下:

使用工具類 Arrays.asList() 把數(shù)組轉(zhuǎn)換成集合時(shí),不能使用其修改集合相關(guān)的方法, 它的 add/remove/clear 方法會(huì)拋出 UnsupportedOperationException 異常。

我在之前的一個(gè)項(xiàng)目中就遇到一個(gè)類似的坑。

Arrays.asList()在平時(shí)開發(fā)中還是比較常見的,我們可以使用它將一個(gè)數(shù)組轉(zhuǎn)換為一個(gè) List 集合。

  1. String[] myArray = {"Apple""Banana""Orange"}; 
  2. List<String> myList = Arrays.asList(myArray); 
  3. //上面兩個(gè)語句等價(jià)于下面一條語句 
  4. List<String> myList = Arrays.asList("Apple","Banana""Orange"); 

JDK 源碼對(duì)于這個(gè)方法的說明:

  1. /** 
  2.   *返回由指定數(shù)組支持的固定大小的列表。此方法作為基于數(shù)組和基于集合的API之間的橋梁, 
  3.   * 與 Collection.toArray()結(jié)合使用。返回的List是可序列化并實(shí)現(xiàn)RandomAccess接口。 
  4.   */ 
  5. public static <T> List<T> asList(T... a) { 
  6.     return new ArrayList<>(a); 

下面我們來總結(jié)一下使用注意事項(xiàng)。

1、Arrays.asList()是泛型方法,傳遞的數(shù)組必須是對(duì)象數(shù)組,而不是基本類型。

  1. int[] myArray = {1, 2, 3}; 
  2. List myList = Arrays.asList(myArray); 
  3. System.out.println(myList.size());//1 
  4. System.out.println(myList.get(0));//數(shù)組地址值 
  5. System.out.println(myList.get(1));//報(bào)錯(cuò):ArrayIndexOutOfBoundsException 
  6. int[] array = (int[]) myList.get(0); 
  7. System.out.println(array[0]);//1 

當(dāng)傳入一個(gè)原生數(shù)據(jù)類型數(shù)組時(shí),Arrays.asList() 的真正得到的參數(shù)就不是數(shù)組中的元素,而是數(shù)組對(duì)象本身!此時(shí) List 的唯一元素就是這個(gè)數(shù)組,這也就解釋了上面的代碼。

我們使用包裝類型數(shù)組就可以解決這個(gè)問題。

  1. Integer[] myArray = {1, 2, 3}; 

2、使用集合的修改方法: add()、remove()、clear()會(huì)拋出異常。

  1. List myList = Arrays.asList(1, 2, 3); 
  2. myList.add(4);//運(yùn)行時(shí)報(bào)錯(cuò):UnsupportedOperationException 
  3. myList.remove(1);//運(yùn)行時(shí)報(bào)錯(cuò):UnsupportedOperationException 
  4. myList.clear();//運(yùn)行時(shí)報(bào)錯(cuò):UnsupportedOperationException 

Arrays.asList() 方法返回的并不是 java.util.ArrayList ,而是 java.util.Arrays 的一個(gè)內(nèi)部類,這個(gè)內(nèi)部類并沒有實(shí)現(xiàn)集合的修改方法或者說并沒有重寫這些方法。

  1. List myList = Arrays.asList(1, 2, 3); 
  2. System.out.println(myList.getClass());//class java.util.Arrays$ArrayList 

下圖是 java.util.Arrays$ArrayList 的簡(jiǎn)易源碼,我們可以看到這個(gè)類重寫的方法有哪些。

  1. private static class ArrayList<E> extends AbstractList<E> 
  2.       implements RandomAccess, java.io.Serializable 
  3.   { 
  4.       ... 
  5.  
  6.       @Override 
  7.       public E get(int index) { 
  8.         ... 
  9.       } 
  10.  
  11.       @Override 
  12.       public E set(int index, E element) { 
  13.         ... 
  14.       } 
  15.  
  16.       @Override 
  17.       public int indexOf(Object o) { 
  18.         ... 
  19.       } 
  20.  
  21.       @Override 
  22.       public boolean contains(Object o) { 
  23.          ... 
  24.       } 
  25.  
  26.       @Override 
  27.       public void forEach(Consumer<? super E> action) { 
  28.         ... 
  29.       } 
  30.  
  31.       @Override 
  32.       public void replaceAll(UnaryOperator<E> operator) { 
  33.         ... 
  34.       } 
  35.  
  36.       @Override 
  37.       public void sort(Comparator<? super E> c) { 
  38.         ... 
  39.       } 
  40.   } 

我們?cè)倏匆幌耲ava.util.AbstractList的 add/remove/clear 方法就知道為什么會(huì)拋出 UnsupportedOperationException 了。

  1. public E remove(int index) { 
  2.     throw new UnsupportedOperationException(); 
  3. public boolean add(E e) { 
  4.     add(size(), e); 
  5.     return true
  6. public void add(int index, E element) { 
  7.     throw new UnsupportedOperationException(); 
  8.  
  9. public void clear() { 
  10.     removeRange(0, size()); 
  11. protected void removeRange(int fromIndex, int toIndex) { 
  12.     ListIterator<E> it = listIterator(fromIndex); 
  13.     for (int i=0, n=toIndex-fromIndex; i<n; i++) { 
  14.         it.next(); 
  15.         it.remove(); 
  16.     } 

那我們?nèi)绾握_的將數(shù)組轉(zhuǎn)換為 ArrayList ?

1、手動(dòng)實(shí)現(xiàn)工具類

  1. static <T> List<T> arrayToList(final T[] array) { 
  2.   final List<T> l = new ArrayList<T>(array.length); 
  3.  
  4.   for (final T s : array) { 
  5.     l.add(s); 
  6.   } 
  7.   return l; 
  8.  
  9.  
  10. Integer [] myArray = { 1, 2, 3 }; 
  11. System.out.println(arrayToList(myArray).getClass());//class java.util.ArrayList 

2、最簡(jiǎn)便的方法

  1. List list = new ArrayList<>(Arrays.asList("a""b""c")) 

3、使用 Java8 的 Stream(推薦)

  1. Integer [] myArray = { 1, 2, 3 }; 
  2. List myList = Arrays.stream(myArray).collect(Collectors.toList()); 
  3. //基本類型也可以實(shí)現(xiàn)轉(zhuǎn)換(依賴boxed的裝箱操作) 
  4. int [] myArray2 = { 1, 2, 3 }; 
  5. List myList = Arrays.stream(myArray2).boxed().collect(Collectors.toList()); 

4、使用 Guava

對(duì)于不可變集合,你可以使用ImmutableList類及其of()與copyOf()工廠方法:(參數(shù)不能為空)

  1. List<String> il = ImmutableList.of("string""elements");  // from varargs 
  2. List<String> il = ImmutableList.copyOf(aStringArray);      // from array 

對(duì)于可變集合,你可以使用Lists類及其newArrayList()工廠方法:

  1. List<String> l1 = Lists.newArrayList(anotherListOrCollection);    // from collection 
  2. List<String> l2 = Lists.newArrayList(aStringArray);               // from array 
  3. List<String> l3 = Lists.newArrayList("or""string""elements"); // from varargs 

5、使用 Apache Commons Collections

  1. List<String> list = new ArrayList<String>(); 
  2. CollectionUtils.addAll(list, str); 

6、 使用 Java9 的 List.of()方法

  1. Integer[] array = {1, 2, 3}; 
  2. List<Integer> list = List.of(array); 

 

責(zé)任編輯:武曉燕 來源: JavaGuide
相關(guān)推薦

2021-06-10 06:59:34

Redis應(yīng)用API

2022-06-21 11:24:05

多線程運(yùn)維

2009-12-17 14:53:52

VS2008程序

2021-05-20 10:02:50

系統(tǒng)Redis技巧

2020-10-22 07:09:19

TCP網(wǎng)絡(luò)協(xié)議

2021-09-11 19:00:54

Intro元素MemoryCache

2021-07-11 09:34:45

ArrayListLinkedList

2024-02-04 08:26:38

線程池參數(shù)內(nèi)存

2024-06-28 10:01:04

2022-10-25 18:00:00

Redis事務(wù)生產(chǎn)事故

2011-08-18 13:49:32

筆記本技巧

2024-09-05 08:07:55

2010-01-06 10:56:47

華為交換機(jī)使用

2019-10-10 15:40:17

redisbug數(shù)據(jù)庫

2020-04-07 08:00:02

Redis緩存數(shù)據(jù)

2021-01-18 11:27:03

Istio架構(gòu)云環(huán)境

2018-08-16 15:11:47

Code ReviewPPT代碼

2019-10-28 08:44:29

Code Review代碼團(tuán)隊(duì)

2015-11-17 16:11:07

Code Review

2022-10-27 10:33:48

敏捷開發(fā)開發(fā)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 国产精品一区二区不卡 | 久久中文字幕一区 | 91精品国产综合久久香蕉麻豆 | 日韩免费av一区二区 | 中文字幕 国产精品 | 久草网址| 亚洲欧美精品在线观看 | 99久久精品免费看国产四区 | 国产亚洲网站 | 三级黄色片在线观看 | 亚洲自拍偷拍免费视频 | 特级做a爱片免费69 精品国产鲁一鲁一区二区张丽 | 欧美第一页| 成人免费一区二区三区视频网站 | 国内精品久久久久久 | 成人在线视频网址 | 最新国产精品视频 | 国产精品有限公司 | 欧洲精品在线观看 | 国产精品一区二区三区在线 | 久久国产精品久久久久久 | 精产国产伦理一二三区 | 亚洲综合一区二区三区 | 国产一卡二卡三卡 | 天天精品综合 | 中文字幕蜜臀av | 在线观看视频h | 91久久综合 | 999久久久精品 | 亚洲国产网址 | 欧美成人一区二区 | 韩国精品在线观看 | 日韩精品网站 | 精品国产黄a∨片高清在线 成人区精品一区二区婷婷 日本一区二区视频 | 日韩精品免费一区二区在线观看 | 九九精品在线 | 亚洲精品视频观看 | 天天射天天干 | 国产a区 | 在线观看成人小视频 | 国产精品欧美一区二区三区不卡 |