array.aslist隐藏的 “坑” 你中招了吗?
作者 | 浩说编程
"Array.asList"
先给不熟悉的兄弟们科普一下:
Array.asList()方法用于将数组转化为集合对于经常做数据处理的朋友来说应该并不陌生不过话说回来这个方法有几个隐藏的"坑"可能有兄弟会中招咱们接下来就一探究竟
01 | 不能直接转换基本数据类型的数组
错误案例:
//定义基本数据类型int类数组
int[] arr = {1, 2, 3};
//使用Array.asList()方法转换为集合
List list = Arrays.asList(arr);
//输出转换之后的集合信息
log.info("list:{} size:{}", list, list.size());
期望输出:
list:[1, 2, 3] size:3
控制台实际输出:
list:[[I@1c53fd30] size:1
很明显隐藏的”坑“出现了,
拥有三个元素的数组在转换之后只剩一个元素且数据类型有问题。
原因分析:
虽然int可以装箱为包装类integer,但int数组则无法整体装箱成integer数组。
脱坑方案:
1、Java8以上提供了Arrays.stream方法强制转换
int[] arr1 = {1, 2, 3};
List list1 = Arrays.stream(arr1).boxed().collect(Collectors.toList());
log.info("list:{} size:{}", list1, list1.size());
2、直接使用包装类integer定义数组
Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
log.info("list:{} size:{}", list2, list2.size());
修改之后的控制台输出:
list:[1, 2, 3] size:3
得到了我们期望的结果,第一个坑我们就脱坑成功!
福利 " 上菜 "
不知不觉又到了经典"上菜"环节
兄弟们准备好了嘛!
今日菜系:
800道大厂面试题
1000道基础面试题
扫码 发送数字 "7"
找面试题再也不用东拼西凑,爽到飞起!
02 | 转换之后的集合无法增删元素
错误案例:
//这次我们用引用类String数组
String[] arr = {"1", "2", "3"};
List list = new ArrayList(Arrays.asList(arr));
try {
//尝试向转换后的list追加元素
list.add("5");
} catch (Exception ex) {
ex.printStackTrace();
}
//转换之后,修改原数组的值
arr[1] = "4";
//输出原数组、转换后的集合
log.info("arr:{} list:{}", Arrays.toString(arr), list);
期望输出:
arr:[1, 4, 3] list:[1, 2, 3, 5]
控制台实际输出:
//首先是list追加元素的异常
java.lang.UnsupportedOperationException
at java.util.AbstractList.add(AbstractList.java:148)
at java.util.AbstractList.add(AbstractList.java:108)
at org.geekbang.time.commonmistakes.collection.aslist.AsListApplication.wrong2(AsListApplication.java:41)
at org.geekbang.time.commonmistakes.collection.aslist.AsListApplication.main(AsListApplication.java:15)
//元素输出
arr:[1, 4, 3] list:[1, 4, 3]
根据控制台输出来看,我们不仅向list追加元素失败,我们对原数组中元素的修改也影响到了集合list,这就是第三个坑点:
03 | 对原始数组的修改会影响转换之后的List
原因分析:
其实Arrays.asList方法返回的List并不是我们期望的java.util.ArrayList,而是Arrays的内部类ArrayList。
两者的区别在于,ArrayList内部类继承自AbstractList类,并没有覆写父类的add方法,所以就产生了上面的异常。
至于第三个坑点则是因为ArrayList直接使用了原始的数组,所以会产生相互共享数组的效果。
如果我们把通过Arrays.asList获得的List交给其他方法处理,很容易因为共享了数组,相互修改产生隐式的“bug”。
这种问题是很难找原因的,要特别小心。
脱坑方案:
方法其实不难猜想,
我们只需要用一个真正的java.util.ArrayList来存放转换之后的list即可
String[] arr = {"1", "2", "3"};
//用java.util.ArrayList接收转换后的list
List list = new ArrayList(Arrays.asList(arr));
arr[1] = "4";
try {
list.add("5");
} catch (Exception ex) {
ex.printStackTrace();
}
log.info("arr:{} list:{}", Arrays.toString(arr), list);
修改之后的控制台输出:
arr:[1, 4, 3] list:[1, 2, 3, 5]
输出符合我们的预期。
我们new的ArrayList既可以做add操作、又和之前的数组实现了分离。
这样就很好的解决了问题!