Java中數(shù)組轉(zhuǎn)List的三種方法與對比分析
前言:
本文介紹Java中數(shù)組轉(zhuǎn)為List三種情況的優(yōu)劣對比,以及應用場景的對比,以及程序員常犯的類型轉(zhuǎn)換錯誤原因解析。
一.最常見方式(未必最佳)
通過 Arrays.asList(strArray)
方式,將數(shù)組轉(zhuǎn)換List后,不能對List增刪,只能查改,否則拋異常。
關鍵代碼:List list = Arrays.asList(strArray);
private void testArrayCastToListError() { String[] strArray = new String[2]; List list = Arrays.asList(strArray); //對轉(zhuǎn)換后的list插入一條數(shù)據(jù) list.add("1"); System.out.println(list); }
執(zhí)行結果:
Exception in thread "main" java.lang.UnsupportedOperationException
at java.util.AbstractList.add(AbstractList.java:148)
at java.util.AbstractList.add(AbstractList.java:108)
at com.darwin.junit.Calculator.testArrayCastToList(Calculator.java:19)
at com.darwin.junit.Calculator.main(Calculator.java:44)
程序在list.add(“1”)
處,拋出異常:UnsupportedOperationException。
原因解析:
Arrays.asList(strArray)
返回值是java.util.Arrays
類中一個私有靜態(tài)內(nèi)部類java.util.Arrays.ArrayList
,它并非java.util.ArrayList
類。java.util.Arrays.ArrayList
類具有 set()
,get()
,contains()
等方法,但是不具有添加add()
或刪除remove()
方法,所以調(diào)用add()
方法會報錯。
使用場景:Arrays.asList(strArray)
方式僅能用在將數(shù)組轉(zhuǎn)換為List后,不需要增刪其中的值,僅作為數(shù)據(jù)源讀取使用。
二.數(shù)組轉(zhuǎn)為List后,支持增刪改查的方式
通過ArrayList的構造器,將Arrays.asList(strArray)
的返回值由java.util.Arrays.ArrayList
轉(zhuǎn)為java.util.ArrayList
。
關鍵代碼:ArrayList<String> list = new ArrayList<String>(Arrays.asList(strArray)) ;
private void testArrayCastToListRight() { String[] strArray = new String[2]; ArrayList<String> list = new ArrayList<String>(Arrays.asList(strArray)) ; list.add("1"); System.out.println(list); }
執(zhí)行結果:成功追加一個元素“1”。
[null, null, 1]
使用場景:需要在將數(shù)組轉(zhuǎn)換為List后,對List進行增刪改查操作,在List的數(shù)據(jù)量不大的情況下,可以使用。
三.通過集合工具類Collections.addAll()方法(最高效)
通過Collections.addAll(arrayList, strArray)
方式轉(zhuǎn)換,根據(jù)數(shù)組的長度創(chuàng)建一個長度相同的List,然后通過Collections.addAll()
方法,將數(shù)組中的元素轉(zhuǎn)為二進制,然后添加到List中,這是最高效的方法。
關鍵代碼:
ArrayList< String> arrayList = new ArrayList<String>(strArray.length); Collections.addAll(arrayList, strArray);
測試:
private void testArrayCastToListEfficient(){ String[] strArray = new String[2]; ArrayList< String> arrayList = new ArrayList<String>(strArray.length); Collections.addAll(arrayList, strArray); arrayList.add("1"); System.out.println(arrayList); }
執(zhí)行結果:同樣成功追加一個元素“1”。
[null, null, 1]
使用場景:需要在將數(shù)組轉(zhuǎn)換為List后,對List進行增刪改查操作,在List的數(shù)據(jù)量巨大的情況下,優(yōu)先使用,可以提高操作速度。
注:附上Collections.addAll()
方法源碼:
public static <T> boolean addAll(Collection<? super T> c, T... elements) { boolean result = false; for (T element : elements) result |= c.add(element);//result和c.add(element)按位或運算,然后賦值給result return result; }
四.Java8可通過stream流將3種基本類型數(shù)組轉(zhuǎn)為List
如果JDK版本在1.8以上,可以使用流stream
來將下列3種數(shù)組快速轉(zhuǎn)為List
,分別是int[]
、long[]
、double[]
,其他數(shù)據(jù)類型比如short[]
、byte[]
、char[]
,在JDK1.8中暫不支持。由于這只是一種常用方法的封裝,不再納入一種嶄新的數(shù)組轉(zhuǎn)List方式,暫時算是java流送給我們的常用工具方法吧。
轉(zhuǎn)換代碼示例如下:
List<Integer> intList= Arrays.stream(new int[] { 1, 2, 3, }).boxed().collect(Collectors.toList()); List<Long> longList= Arrays.stream(new long[] { 1, 2, 3 }).boxed().collect(Collectors.toList()); List<Double> doubleList= Arrays.stream(new double[] { 1, 2, 3 }).boxed().collect(Collectors.toList());
如果是String數(shù)組,可以使用Stream
流這樣轉(zhuǎn)換:
String[] arrays = {"tom", "jack", "kate"}; List<String> stringList= Stream.of(arrays).collect(Collectors.toList());
--------------------------------------正文結束,以下是對評論中的問題集中回復和講解--------------------------------------
補充:回答評論中的疑問
問題: 有評論提出:數(shù)組類型如果是整型數(shù)組,轉(zhuǎn)為List時,會報錯?
答案: 在JDK1.8
環(huán)境中測試,這三種轉(zhuǎn)換方式是沒有問題的。放心使用。對于Integer[]
整型數(shù)組轉(zhuǎn)List的方法和測試結果如下:
- 方式一:不支持增刪
Integer[] intArray1 = new Integer[2]; List<Integer> list1 = Arrays.asList(intArray1); System.out.println(list1);
運行結果:
[null, null]
- 方式二:支持增刪
Integer[] intArray2 = new Integer[2]; List<Integer> list2 = new ArrayList<Integer>(Arrays.asList(intArray2)) ; list2.add(2); System.out.println(list2);
運行結果:
[null, null, 2]
- 方式三:支持增刪,且數(shù)據(jù)量大最高效
Integer[] intArray3 = new Integer[2]; List<Integer> list3 = new ArrayList<Integer>(intArray3.length); Collections.addAll(list3, intArray3); list3.add(3); System.out.println(list3);
運行結果:
[null, null, 3]
綜上,整型Integer[]
數(shù)組轉(zhuǎn)List<Integer>
的正確方式應該是這樣的。
猜想你們遇到的問題: 由于評論沒有給出報錯的代碼,所以我猜想你們出現(xiàn)的錯誤可能是這樣轉(zhuǎn)換的:
int[] intArray1 = new int[2]; List<Integer> list1 = Arrays.asList(intArray1);//此處報錯?。?!
報錯原因:等號兩邊類型不一致,當然編譯不通過。分析見下文。
那么在聲明數(shù)組時,用int[]
還是Integer[]
,哪種聲明方式才能正確的轉(zhuǎn)為List
呢?
答案: 只能用Integer[]
轉(zhuǎn)List<Integer>
,即只能用基本數(shù)據(jù)類型的包裝類型,才能直接轉(zhuǎn)為List
。
原因分析如下:
我們來看List
在Java源碼中的定義(別害怕看不懂源碼,看我分析,很易懂的):
public interface List<E> extends Collection<E> {省略…}
再來看Arrays.asList()
的在Java源碼定義:
public static <T> List<T> asList(T... a) { return new ArrayList<>(a); }
從上述源碼中可以看出,
List
聲明時,需要傳遞一個泛型<E>
作為形參,asList()
參數(shù)類型也是泛型中的通配類型<T>
。Java中所有的泛型必須是引用類型。什么是引用類型?
Integer
是引用類型,那int
是什么類型?int
是基本數(shù)據(jù)類型,不是引用類型。這就是為什么java中沒有List<int>
,而只有List<Integer>
。舉一反三:其他8種基本數(shù)據(jù)類型
byte、short、int、long、float、double、char
也都不是引用類型,所以8種基本數(shù)據(jù)類型都不能作為List的形參。但String、數(shù)組、class、interface
是引用類型,都可以作為List的形參,所以存在List<Runnable>
接口類型的集合、List<int[]>
數(shù)組類型的集合、List<String>
類的集合。但不存在list<byte>
、list<short>
等基本類型的集合。
有了上述基礎知識后,再來看為什么下面兩行代碼第二行能編譯通過,第三行卻編譯報錯?
int[] intArray1 = new int[1]; Arrays.asList(intArray1);//編譯不報錯 List<Integer> list1 = Arrays.asList( intArray1);//編譯報錯
答案:
- 第二行代碼,
Arrays.asList()
方法的入?yún)⑹莻€引用類型的int[]
,那么返回值類型一定是List<int[]>
,其完整代碼是:List<int[]> intsArray = Arrays.asList(intArray1);
,所以編譯通過,沒問題。 - 第三行報錯,因為等號兩邊的類型不一致,左邊:
List<Integer>
,右邊List<int[]>
,所以編譯時就報錯。
總結
現(xiàn)在你應該明白,為什么int[]
不能直接轉(zhuǎn)換為List<Integer>
,而Integer[]
就可以轉(zhuǎn)換為List<Integer>
了吧。因為List
中的泛型必須是引用類型,int
是基本數(shù)據(jù)類型,不是引用類型,但int
的包裝類型Integer
是class
類型,屬于引用類型,所以Integer
可以作為List
形參,List<Integer>
在java中是可以存在的,但不存在List<int>
類型。
在編碼時,我們不光要知其然,還要知其所以然,通過分析JDK源碼,才能得出一手信息,不僅了解到了如何用,還能得出為何這樣用。
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
基于Tomcat7、Java、WebSocket的服務器推送聊天室實例
HTML5 WebSocket實現(xiàn)了服務器與瀏覽器的雙向通訊,本篇文章主要介紹了基于Tomcat7、Java、WebSocket的服務器推送聊天室實例,具有一定的參考價值,有興趣的可以了解一下。2016-12-12java-collection中的null,isEmpty用法
這篇文章主要介紹了java-collection中的null,isEmpty用法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02Java如何使用Agent和ASM在字節(jié)碼層面實現(xiàn)方法攔截
Agent是一種運行在 Java 虛擬機 (JVM) 上的特殊程序,ASM是一個輕量級的 Java 字節(jié)碼編輯和分析框架,本文為大家介紹了如何利用他們在字節(jié)碼層面實現(xiàn)方法攔截,感興趣的可以了解一下2023-05-05使用Java構造和解析Json數(shù)據(jù)的兩種方法(詳解二)
這篇文章主要介紹了使用Java構造和解析Json數(shù)據(jù)的兩種方法(詳解二)的相關資料,需要的朋友可以參考下2016-03-03