内容简介:使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合。而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比较更方便的打印函数Arrays.toString(),于是打印不再使用asList(),而asList()恰巧可用于将数组转为集合。
使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合。
而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比较更方便的打印函数Arrays.toString(),于是打印不再使用asList(),而asList()恰巧可用于将数组转为集合。
1
错误用法
如果你这样使用过,那你可要注意了。
-
错误一
将基本类型数组作为asList的参数
int[] arr = {1,2,3}; List list = Arrays.asList(arr); System.out.println(list.size());
猜一下输出结果?
-
错误二
将数组作为asList参数后,修改数组或List
String[] arr = {"欢迎","关注","Java"}; List list = Arrays.asList(arr); arr[1] = "爱上"; list.set(2,"我"); System.out.println(Arrays.toString(arr)); System.out.println(list.toString());
猜一下输出结果?
-
错误三
数组转换为集合后,进行增删元素
String[] arr = {"欢迎","关注","Java"}; List list = Arrays.asList(arr); list.add("新增"); list.remove("关注"); System.out.println(list.toString());
猜一下输出结果?
你是不是以为上面 :point_up_2:那个 list 是 java.util.ArrayList ?
答案很确定: NO !
2
深入探究
我们通过asList()源码可发现其原因,但为了更直观,我们先通过IDEA debug来看看结果。
List<String> asList = Arrays.asList("欢迎","关注","码上实战"); ArrayList<String> aList = new ArrayList<>(asList);
其实它返回的是 java.util.Arrays.ArrayList ,这个家伙是谁呢?
请看下源码:
public class Arrays { //省略其他方法 public static <T> List<T> asList(T... a) { return new ArrayList<>(a); } //就是这个家伙 :point_down: private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, java.io.Serializable{ private final E[] a; ArrayList(E[] array) { a = Objects.requireNonNull(array); } @Override public int size() { return a.length; } //省略其他方法 } }
但它和ArrayList貌似很像唉!有什么不同吗?
3
不同之处
Arrays.ArrayList 是 工具 类 Arrays 的一个内部静态类,它没有完全实现List的方法,而 ArrayList直接实现了List 接口,实现了List所有方法。
-
长度不同 和 实现的方法不同
Arrays.ArrayList是一个定长集合,因为它没有重写add,remove方法,所以一旦初始化元素后,集合的size就是不可变的。
-
参数赋值方式不同
Arrays.ArrayList将外部数组的引用直接通过“=”赋予内部的泛型数组,所以本质指向同一个数组。
ArrayList(E[] array) { a = array; }
ArrayList是将其他集合转为数组后copy到自己内部的数组的。
public ArrayList(Collection<? extends E> c) { // toArray 底层使用的是 数组clone 或 System.arraycopy elementData = c.toArray(); }
4
揭晓答案
-
错误一
由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] arr数组当成了一个泛型对象,所以集合中最终只有一个元素arr。
-
错误二
由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编码时可能产生莫名的问题。
-
错误三
由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。
5
支持基础类型的方式
-
如果使用Spring
int[] a = {1,2,3}; List list = CollectionUtils.arrayToList(a); System.out.println(list);
-
如果使用 Java 8
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
6
数组转ArrayList
-
遍历转换
Integer intArray[] = {1, 2, 3}; ArrayList<Integer> aList = new ArrayList<>(); for (Integer i: intArray){ aList.add(i); }
显然这种方式不够优雅!反正我不愿意使用。
-
使用工具类
上面方案不够优雅,那么这种相对来说优雅一些。
List<String> list = new ArrayList(); Collections.addAll(list, "welcome", "to", "china");
你以为这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。
-
如果使用Java 8
既可以用于基本类型也可以返回想要的集合。
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
-
两个集合类结合
将Arrays.asList返回的集合作为ArrayList的构造参数
ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));
7
最后
勿以点小而不闻!体现程序素养或许就在这些小地方,不要给自己或别人留坑。
那么这个知识点,你get到了吗?get到了,那来继续关注我。没get到?来来来,咱俩单独聊聊。
以上所述就是小编给大家介绍的《千万不要这样使用 Arrays.asList》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- RecyclerView使用指南(一)—— 基本使用
- 如何使用Meteorjs使用URL参数
- 使用 defer 还是不使用 defer?
- 使用 Typescript 加强 Vuex 使用体验
- [译] 何时使用 Rust?何时使用 Go?
- UDP协议的正确使用场合(谨慎使用)
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Java并发编程实战
Brian Goetz、Tim Peierls、Joshua Bloch、Joseph Bowbeer、David Holmes、Doug Lea / 童云兰 / 机械工业出版社华章公司 / 2012-2 / 69.00元
本书深入浅出地介绍了Java线程和并发,是一本完美的Java并发参考手册。书中从并发性和线程安全性的基本概念出发,介绍了如何使用类库提供的基本并发构建块,用于避免并发危险、构造线程安全的类及验证线程安全的规则,如何将小的线程安全类组合成更大的线程安全类,如何利用线程来提高并发应用程序的吞吐量,如何识别可并行执行的任务,如何提高单线程子系统的响应性,如何确保并发程序执行预期任务,如何提高并发代码的性......一起来看看 《Java并发编程实战》 这本书的介绍吧!
正则表达式在线测试
正则表达式在线测试
HEX CMYK 转换工具
HEX CMYK 互转工具