2. 数组与list互转
实战2:数组与list互转
这个考题比较常见,也比较简单,难道就这也有什么可以说到的门路不成?
接下来本文好好的说一说它的几种实现姿势,总有一款你喜欢的
1.数组转List
1.1. Array.asList
这个考题太简单了,直接使用Array.asList
不就完事了么,比如
@Test
public void ary2list() {
String[] ary = new String[]{ "1", "a"};
List<String> list = Arrays.asList((ary);
System.out.println(list);
}
数组转list,so easy!!!
真的就这么简单么???
且看下面这一段代码
public void ary2list() {
String[] ary = new String[]{ "1", "a"};
List<String> list = Arrays.asList((ary);
System.out.println(list);
list.add("c");
System.out.println(list);
}
直接抛出了异常java.lang.UnsupportedOperationException
有兴趣的小伙伴可以看一下源码实现方式,通过Arrays.asList
创建的List,虽说也命名是ArrayList
,但是它的全路径为 java.util.Arrays.ArrayList
, 不支持add
, remove
等操作(所以下次再有面试官问ArrayList的知识点时,就可以反问一句,老哥你指的是哪个ArrayList😝,逼格是不是立马拉起来)
知识点
- 通过
Arrays.asList
创建的列表,不允许新增,删除元素;但是可以更新列表中元素的值
1.2. new ArrayList
上面的数组转list方式虽然是最简单的,但不一定是合适的,特别是当我们可能对转换后的list进行操作时,可能埋坑(而且这种坑还非常隐晦,代码层面上很难发现)
为了减少在代码里面下毒的可能性,不妨使用下面这种方式new ArrayList<>(Arrays.asList(ary))
String[] ary = new String[]{ "1", "a"};
List<String> out = new ArrayList<>(Arrays.asList(ary));
out.add("hello");
System.out.println(out);
通过上面这种方式创建的List,就是我们熟知的ArrayList
了
避雷预警
看到上面这个使用姿势,就很容易想到一个常见的踩雷点,比如我们的应用中,有一个全局共享的配置列表,张三需要拿id为奇数的配置,李四拿id为偶数的配置,然后他们都是这么做的
list.removeIf(s -> s.id % 2 == 0);
然后跑了一次之后发现这个全局的列表清空了,这就是典型的没有做好资源隔离的case了,针对这种场景,要么是限制使用方,直接针对全局的资源进行修改,要么就是使用方拿到的是一个隔离的备份
禁止修改:
- 使用不可变的容器,如前面提到的
java.util.Arrays.ArrayList
() - 使用
Collections.unmodifiableList
创建
List<String> unModifyList = Collections.unmodifiableList(out);
列表拷贝
new ArrayList<>(Arrays.asList(ary));
(上面这种属于深拷贝的实现,具体可以看一下jdk的源码实现)
1.3. Collections.addAll
第三种方式借助jdk提供的容器工具类Collections
来实现
@Test
public void ary2listV3() {
String[] ary = new String[]{ "1", "a"};
// 创建列表,并指定长度,避免可能产生的扩容
List<String> out = new ArrayList<>(ary.length);
// 实现数组添加到列表中
Collections.addAll(out, ary);
// 因为列表为我们定义的ArrayList,因此可以对它进行增删改
out.add("hello");
System.out.println(out);
}
原则上是比较推荐这种方式来实现的,至于为啥?看下源码实现
public static <T> boolean addAll(Collection<? super T> c, T... elements) {
boolean result = false;
for (T element : elements)
result |= c.add(element);
return result;
}
这段代码的实现是不是非常眼熟,如果让我们自己来写,也差不多会写成这样吧,简单直观高效,完美
2. 列表转数组
不同于数组转列表的几种玩法,列表转数组就简单多了,直接调用List.toArray
List<String> list = Arrays.asList("a", "b", "c");
// 返回的是Object[] 数组
Object[] cell = list.toArray();
// 如果需要指定数组类型,可以传一个指定各类型的空的数组
// 也可以传一个与目标列表长度相等的数组,这样会将列表中的元素拷贝到这个数组中
String[] strCell = list.toArray(new String[]{});
3. 小结
今天的博文主题是数组与列表的互转,虽说题目简单,但是实现方式也是多种,需要搞清楚它们之间的本质区别,一不小心就可能采坑,而最简单的地方掉坑里,往往是最难发现和爬出来的
核心知识点小结如下
数组转list:
Arrays.asList(xxx)
:创建的是不可变列表,不能删除和新增元素new ArrayList<>(Arrays.asList(xxx)
: 相当于用列表创建列表,属于深拷贝的一种表现,获取到的列表支持新增、删除- 推荐写法
Collections.addAll()
列表转数组
list.toArray
: 如果需要指定数组类型,则传参指定