千万别这样使用Arrays.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());
猜一下输出结果?
你是不是以为上面 ????那个 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); } //就是这个家伙 ???? 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到?来来来,咱俩单独聊聊。
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。