全是坑!慎用 Arrays.asList

飞天小牛肉

共 1880字,需浏览 4分钟

 · 2024-04-10

面试题库 自研短链项目 简历修改&模拟面试 招聘信息


Java 8 提供的 Stream 流式处理大大减少了集合类各种操作(投影、过滤、转换)的代码量,用起来非常香,所以在实际业务开发中,我们常常会把原始的数组转换为 List 类数据结构,使得其可以用上 Stream 流操作。

Arrays.asList 方法应该是各位最常用的数组一键转换为 List 的方法了,但这个方法有几个坑,如果不了解的话,排查 Bug 可能会比较困难:

  1. 坑 1:不能直接使用 Arrays.asList 来转换基本类型数组
  2. 坑 2:Arrays.asList 返回的 List 不支持增删操作
  3. 坑 3:对原始数组的修改会影响 Arras.asList 返回的那个 List

第一个坑

在如下代码中,我们初始化三个数字的 int[]数组,然后使用 Arrays.asList 把数组转换为 List:

72177bef5300a0d6160a717a2a3ef72d.webp

但,这样初始化的 List 并不是我们期望的包含 3 个数字的 List,输出结果如下:

995621cf098ed8b5fe4069c00223a40c.webp可以发现,这个 List 包含的其实是一个 int 数组,整个 List 的元素个数是 1,元素类型是整数数组

其原因是,只能是把单个 int 类型装箱为 Integer,不能把 int 数组装箱为 Integer 数组。Arrays.asList 方法传入的是一个泛型 T 类型可变参数,所以 int 数组实际上是被整体看成一个对象作为泛型类型 T:

6087f0bab8173611cba88a2aae0c7b45.webp

以上,就是第一个坑:不能直接使用 Arrays.asList 来转换基本类型数组。直接遍历这样的 List 必然会出现 Bug,修复方式有两种:

  1. 最简单的,直接把数组声明为包装类型,不要用 int 这种基本类型

    28b2965a6a31caa83c3e62fb3210a07b.webp
  2. 如果使用 Java8 以上版本可以使用 Arrays.stream 方法来转换,stream 流提供了 boxed 装箱操作:

    784b0177979a320e3b1dfdeb9527ab40.webp

第二个坑

把三个字符串 1、2、3 构成的字符串数组,使用 Arrays.asList 转换为 List 后,然后为 List 增加一个字符串 4:

1c571bf595d3dd8c0f974ae032f5ab9b.webp

结果如下,为 List 新增字符串 4 的操作失败了,报错  UnsupportedOperationException

413ec554d78e406cbae431f363f17e9f.webp

第二个坑:Arrays.asList 返回的 List 不支持增删操作

因为 Arrays.asList 返回的 List 并不是我们期望的 java.util.ArrayList,而是 Arrays 的内部类 ArrayList:

670b12f8edafd923c819f4ae887a1db9.webp

ArrayList 内部类继承自 AbstractList 类,并没有覆写父类的 add 方法,而父类中 add 方法的实现,就是抛出 UnsupportedOperationException

26dc43efd4f07234caf7d8dbe29e0494.webp

第三个坑

把三个字符串 1、2、3 构成的字符串数组,使用 Arrays.asList 转换为 List 后,然后修改原字符串数组的第一个元素为 0:

73dcf61c5654687df13717ff279a5261.webp

输出如下,把原始数组的第一个元素从 1 修改为 0 后,asList 获得的 List 中的第一个元素也被修改为 0 了:

a43bc76dda9d1a33acc7c707a31fe2c2.webp

第三个坑:对原始数组的修改会影响 Arras.asList 返回的那个 List

看一下 ArrayList 的实现,可以发现 ArrayList 其实是直接使用了原始的数组。所以,我们要特别小心,把通过 Arrays.asList 获得的 List 交给其他方法处理,很容易因为共享数组,相互修改而产生 Bug。

2663be58661ce446e5add0ce01f96608.webp

第二个坑和第三个坑的本质原因是一样的,都是由于数组共享导致的问题,所以解决方式比较简单,重新 new 一个 ArrayList 初始化 Arrays.asList 返回的 List 即可:

dfa3e9305d191d20e8d0048e38fe6773.webp

修改后的代码实现了原始数组和 List 的解耦,不再相互影响。同时,因为操作的是真正的 ArrayList,add 也不再报错:

e59b9537a7872b35bd2562f3060ad56d.webp




以上内容选自我的面试小程序 编程满天星 ,收录真实大厂面经,整理热度排序,提供全网唯一小而美的自研短链项目教程,一对一简历修改 & 模拟面试服务,并定期更新招聘信息,真一站式准备大厂面试!

面试题库 自研短链项目 简历修改&模拟面试 招聘信息


浏览 2
点赞
评论
收藏
分享

手机扫一扫分享

举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

举报