小心!"数组"转"集合"的这几个隐藏"bug"

2021-12-24 19:46:05 浏览数 (1)

先给不熟悉的兄弟们科普一下:

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

0 人点赞