先给不熟悉的兄弟们科普一下:
Array.asList()方法用于将数组转化为集合
对于经常做数据处理的朋友来说
应该并不陌生
不过话说回来
这个方法有几个隐藏的"坑"
可能有兄弟会中招
咱们接下来就一探究竟
听说点赞分享效果更佳呢!!
坑一:不能直接转换基本数据类型的数组
错误案例:
代码语言:javascript复制//定义基本数据类型int类数组
int[] arr = {1, 2, 3};
//使用Array.asList()方法转换为集合
List list = Arrays.asList(arr);
//输出转换之后的集合信息
log.info("list:{} size:{}", list, list.size());
期望输出:
代码语言:javascript复制list:[1, 2, 3] size:3
控制台实际输出:
代码语言:javascript复制list:[[I@1c53fd30] size:1
很明显隐藏的”坑“出现了,
拥有三个元素的数组在转换之后只剩一个元素且数据类型有问题。
原因分析:
虽然int可以装箱为包装类integer,但int数组则无法整体装箱成integer数组。
脱坑方案:
1、Java8以上提供了Arrays.stream方法强制转换
代码语言:javascript复制int[] arr1 = {1, 2, 3};
List list1 = Arrays.stream(arr1).boxed().collect(Collectors.toList());
log.info("list:{} size:{}", list1, list1.size());
2、直接使用包装类integer定义数组
代码语言:javascript复制Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
log.info("list:{} size:{}", list2, list2.size());
修改之后的控制台输出:
代码语言:javascript复制list:[1, 2, 3] size:3
得到了我们期望的结果,第一个坑我们就脱坑成功啦!
兄弟不要吝啬你的点赞哦!是我继续输出的动力,我们继续脱坑:
坑二:转换之后的集合无法增删元素
错误案例:
代码语言:javascript复制//这次我们用引用类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);
期望输出:
代码语言:javascript复制arr:[1, 4, 3] list:[1, 2, 3, 5]
控制台实际输出:
代码语言:javascript复制//首先是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,
这就是第三个坑点:
坑三:对原始数组的修改会影响转换之后的List
原因分析:
其实Arrays.asList方法返回的List并不是我们期望的java.util.ArrayList,而是Arrays的内部类ArrayList。
两者的区别在于,ArrayList内部类继承自AbstractList类,并没有覆写父类的add方法,所以就产生了上面的异常。
至于第三个坑点则是因为ArrayList直接使用了原始的数组,所以会产生相互共享数组的效果。
如果我们把通过Arrays.asList获得的List交给其他方法处理,很容易因为共享了数组,相互修改产生隐式的“bug”。
这种问题是很难找原因的,要特别小心。
脱坑方案:
方法其实不难猜想,
我们只需要用一个真正的java.util.ArrayList来存放转换之后的list即可
代码语言:javascript复制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);
修改之后的控制台输出:
代码语言:javascript复制arr:[1, 4, 3] list:[1, 2, 3, 5]
输出符合我们的预期。
我们new的ArrayList既可以做add操作、又和之前的数组实现了分离。
这样就很好的解决了问题!
end