Java 踩坑记系列之 Arrays.AsList
java.util.Arrays
的asList
方法可以方便的将数组转化为集合,我们平时开发在初始化ArrayList
时使用的比较多,可以简化代码,但这个静态方法asList()
有几个坑需要注意:
一. 如果对集合使用增加或删除元素的操作将会报错
如下代码:
输出结果:
Arrays.asList
的源码虽然返回的是一个ArrayList
,但这个ArrayList
是Arrays内部的一个静态类(java.util.Arrays.ArrayList
),并不是我们平时用的java.util.ArrayList
只是名字一样,包名不一样:
这个ArrayList
也继承自AbstractList
,但是却没有实现add()
,remove()
方法用这两个方法其实调用的是AbstractList
的默认方法:
直接抛出" UnsupportedOperationException
"的异常!
所以当我们调用这个ArrayList
的add
方法时其实是调用了抽象集合类里的add
方法,就抛出了上面的异常,原因是数组虽然提供了转集合的方法, 但本质上还是个数组,数组的长度是固定的,所以jdk的作者没有实现add()
和remove()
方法。
大家在使用的时候需要注意,要结合具体业务场景判断:如果只是对转成后的集合进行遍历或使用stream()流操作都没有问题,可以正常使用(修改元素也没问题) 但如果需要新增元素或删除元素等改变集合长度的操作就要避免,否则报错就会影响到我们正常的业务逻辑。
二. 长度问题
还是Arrays
的asList()
,该方法无法将一个基本类型的数组转换成集合List,或者说基本类型虽然能转换成功,但不是"我们想要的集合",因为asList
方法接收的是一个泛型的变长参数。
而基本数据类型是无法被泛型化的,这样的话对于泛型而言,基本数据类型会被作为一个"[x"的类型,"["表示这是一个数组,"x"为当前数组的类型,例如: int[] 实际上它的类型是"[I"
这样的话Arrays.asList
传入的参数是基本类型的话,会把数组当做一个元素处理(长度=1),如果我们要遍历List
或获取List
的长度就不准确了,类似下面的用法:
输出结果:
如果我们的代码里用到了这样获取长度或判断是否包含指定元素的操作就要注意写法,否则就会影响我们的逻辑流程!
为避免这样的问题出现,有很多种其他写法,比如将array转成包装类(即引用类型),如下:
输出结果:
当然也可以使用google的集合类"Ints.asList",如下:
输出结果:
但是google的Ints.asList
方法返回的list也是一个自己实现的集合,同样不支持add
和remove
方法,如果既要保证转换成list后的数据长度正确又要能增删操作的话,就用java.util.ArrayList吧。
三. Arrays.asList方法返回的是数组的一个视图,也就是说对这个list的操作都会反映在原数组上
评论