Arrays.asList()的坑

今天尝试将int数组转化成list,发现得到的并不是想要的结果,上网查了一下,发现了以前没注意的一个问题。
下面是原文

前言

使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合。

一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比较更方便的打印函数Arrays.toString(),于是打印不再使用asList(),而asList()恰巧可用于将数组转为集合。

错误用法

  • 错误一
int[] arr = {1,2,3};

List list = Arrays.asList(arr);

System.out.println(list.size());
  • 错误二

将数组作为asList参数后,修改数组或List

String[] arr = {"欢迎","关注","Java"};

List list = Arrays.asList(arr);

 

arr[1] = "爱上";

list.set(2,"我");

 

System.out.println(Arrays.toString(arr));

System.out.println(list.toString());
  • 错误三

数组转换为集合后,进行增删元素

String[] arr = {"欢迎","关注","Java"};

List list = Arrays.asList(arr);

 

list.add("新增");

list.remove("关注");

System.out.println(list.toString());

猜一下输出结果?

你是不是以为上面 👆那个 listjava.util.ArrayList ?

答案很确定:NO !

2、深入探究

我们通过asList()源码可发现其原因,但为了更直观,我们先通过IDEA debug来看看结果。

代码:

List<String> asList = Arrays.asList("欢迎","关注","Java");
 
ArrayList<String> aList = new ArrayList<>(asList);

在这里插入图片描述
其实它返回的是 java.util.Arrays.ArrayList,这个家伙是谁呢?

请看下源码:

public class Arrays {
 //省略其他方法
 public static <T> List<T> asList(T... a) {
 return new ArrayList<>(a);
 }

 //就是这个家伙  👇
 private static class ArrayList<E> extends AbstractList<E>
  implements RandomAccess, java.io.Serializable{
 private final E[] a;

 ArrayList(E[] array) {
  a = Objects.requireNonNull(array);
 }

 @Override
 public int size() {
  return a.length;
 }
 //省略其他方法
 }

}

3、不同之处

  • Arrays.ArrayList 是工具类 Arrays 的一个内部静态类,它没有完全实现List的方法,而 ArrayList直接实现了List 接口,实现了List所有方法。
    在这里插入图片描述

  • 长度不同 和 实现的方法不同

Arrays.ArrayList是一个定长集合,因为它没有重写add,remove方法,所以一旦初始化元素后,集合的size就是不可变的。

  • 参数赋值方式不同

Arrays.ArrayList将外部数组的引用直接通过“=”赋予内部的泛型数组,所以本质指向同一个数组。

ArrayList(E[] array) {
 a = array;
}

ArrayList是将其他集合转为数组后copy到自己内部的数组的。

public ArrayList(Collection<? extends E> c) {
 // toArray 底层使用的是 数组clone 或 System.arraycopy
 elementData = c.toArray();
}

4、揭晓答案

  • 错误一

由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] arr数组当成了一个泛型对象,所以集合中最终只有一个元素arr。

第一个例子的输出:

1
  • 错误二

由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编程时可能产生莫名的问题。

第二个例子的输出:

[欢迎, 爱上,]
[欢迎, 爱上,]
  • 错误三

由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。

第三个例子的输出:

Exception in thread "main" java.lang.UnsupportedOperationException
	at java.util.AbstractList.add(AbstractList.java:148)
	at java.util.AbstractList.add(AbstractList.java:108)
	at Test.main(Test.java:31)

5、支持基础类型的方式

  • 如果使用Spring
int[] a = {1,2,3};
List list = CollectionUtils.arrayToList(a);
System.out.println(list);
  • 如果使用Java 8
int intArray[] = {1, 2, 3};
List<Integer> iList = Arrays.stream(intArray)
       .boxed()
       .collect(Collectors.toList());
System.out.println(iList);

6、数组转ArrayList

那怎么把基础类型数组转成ArrayList呢?

  • 遍历转换
Integer intArray[] = {1, 2, 3};
ArrayList<Integer> aList = new ArrayList<>();
for (Integer i: intArray){
 aList.add(i);
}

显然这种方式不够优雅!反正我不愿意使用。

  • 使用工具类

上面方案不够优雅,那么这种相对来说优雅一些。

List<String> list = new ArrayList();
Collections.addAll(list, "welcome", "to", "china");

你以为这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。

  • 如果使用Java 8

既可以用于基本类型也可以返回想要的集合。

int intArray[] = {1, 2, 3};
List<Integer> iList = Arrays.stream(intArray)
       .boxed()
       .collect(Collectors.toList());
System.out.println(iList);
  • 两个集合类结合

将Arrays.asList返回的集合作为ArrayList的构造参数

ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));

7、最后

勿以点小而不闻!体现程序素养或许就在这些小地方,不要给自己或别人留坑。

原文链接:https://mp.weixin.qq.com/s/iDpx7krEjSue0-IXGiMFeQ

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值