Arrays.asList()踩坑记
使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合。
而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比较更方便的打印函数Arrays.toString(),于是打印不再使用asList(),而asList()恰巧可用于将数组转为集合。
1、错误用法
错误一
将基本类型数组作为asList的参数
1 2 3 4 5 |
|
猜一下输出结果?
错误二
将数组作为asList参数后,修改数组或List
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
错误三
数组转换为集合后,进行增删元素
1 2 3 4 5 6 7 8 9 10 11 |
|
2、深入探究
我们通过asList()源码可发现其原因,但为了更直观,我们先通过IDEA debug来看看结果。
1 2 3 |
|
请看下源码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
|
3、不同之处
Arrays.ArrayList 是工具类 Arrays 的一个内部静态类,它没有完全实现List的方法,而 ArrayList直接实现了List 接口,实现了List所有方法。
长度不同 和 实现的方法不同
Arrays.ArrayList是一个定长集合,因为它没有重写add,remove方法,所以一旦初始化元素后,集合的size就是不可变的。
参数赋值方式不同
Arrays.ArrayList将外部数组的引用直接通过“=”赋予内部的泛型数组,所以本质指向同一个数组。
1 2 3 |
|
ArrayList是将其他集合转为数组后copy到自己内部的数组的。
1 2 3 4 |
|
4、揭晓答案
错误一
由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] arr数组当成了一个泛型对象,所以集合中最终只有一个元素arr。
错误二
由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编码时可能产生莫名的问题。
错误三
由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。
5、支持基础类型的方式
如果使用Spring
1 2 3 |
|
如果使用Java 8
1 2 3 4 5 |
|
6、数组转ArrayList
遍历转换
1 2 3 4 5 |
|
显然这种方式不够优雅!反正我不愿意使用。
使用工具类
上面方案不够优雅,那么这种相对来说优雅一些。
1 2 |
|
你以为这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。
如果使用Java 8
既可以用于基本类型也可以返回想要的集合。
1 2 3 4 5 |
|
两个集合类结合
将Arrays.asList返回的集合作为ArrayList的构造参数
1 |
|