原创

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操作、又和之前的数组实现了分离。

    这样就很好的解决了问题!

正文到此结束
本文目录