本站消息

站长简介/公众号

  出租广告位,需要合作请联系站长


+关注
已关注

分类  

暂无分类

标签  

暂无标签

日期归档  

2024-11(1)

千万不要这样使用 Arrays.asList !

发布于2021-05-29 22:12     阅读(1465)     评论(0)     点赞(5)     收藏(2)


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

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

错误用法

如果你这样使用过,那你要注意下了。

  • 错误一

将基本类型数组作为asList的参数

  1. int[] arr = {1,2,3};
  2. List list = Arrays.asList(arr);
  3. System.out.println(list.size());

猜一下输出结果?

  • 错误二

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

  1. String[] arr = {"欢迎","关注","Java"};
  2. List list = Arrays.asList(arr);
  3. arr[1] = "爱上";
  4. list.set(2,"我");
  5. System.out.println(Arrays.toString(arr));
  6. System.out.println(list.toString());

猜一下输出结果?

  • 错误三

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

  1. String[] arr = {"欢迎","关注","Java"};
  2. List list = Arrays.asList(arr);
  3. list.add("新增");
  4. list.remove("关注");

猜一下输出结果?

你是不是以为上面 那个list是 java.util.ArrayList ?

答案很确定:NO!

探索真理

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

  1. List<String> asList = Arrays.asList("欢迎","关注","码上实战");
  2. ArrayList<String> aList = new ArrayList<>(asList);

 

其实它返回的是 java.util.Arrays.ArrayList ,这个家伙是谁呢?

 

请看下源码:

  1. public class Arrays {
  2. //省略其他方法
  3. public static <T> List<T> asList(T... a) {
  4. return new ArrayList<>(a);
  5. }
  6. //就是这个家伙
  7. private static class ArrayList<E> extends AbstractList<E>
  8. implements RandomAccess, java.io.Serializable{
  9. private final E[] a;
  10. ArrayList(E[] array) {
  11. a = Objects.requireNonNull(array);
  12. }
  13. @Override
  14. public int size() {
  15. return a.length;
  16. }
  17. //省略其他方法
  18. }
  19. }

但它和ArrayList貌似很像唉!有什么不同吗?

不同之处

Arrays.ArrayList 是工具类 Arrays 的一个内部静态类,它没有完全实现List的方法,而 ArrayList直接实现了List 接口,实现了List所有方法。

 

  • 长度不同 和 实现的方法不同 Arrays.ArrayList是一个定长集合,因为它没有重写add,remove方法,所以一旦初始化元素后,集合的size就是不可变的。
  • 参数赋值方式不同

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

  1. ArrayList(E[] array) {
  2. a = array;
  3. }

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

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

揭晓答案

  • 错误一 由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] arr数组当成了一个泛型对象,所以集合中最终只有一个元素arr.
  • 错误二 由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编码时可能产生莫名的问题。
  • 错误三 由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。

支持基础类型的方式

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

数组转为ArrayList

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

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

  • 使用工具类 上面方案不够优雅,那么这种相对来说优雅一些。
  1. List<String> list = new ArrayList();
  2. Collections.addAll(list, "welcome", "to", "china");
你以为这种还不错? too young too simple! addAll()方法的实现就是用的上面遍历的方式。
  • 如果使用Java8 既可以用于基本类型也可以返回想要的集合。
  1. int intArray[] = {1, 2, 3};
  2. List<Integer> iList = Arrays.stream(intArray)
  3. .boxed()
  4. .collect(Collectors.toList());
  5. System.out.println(iList);
  • 两个集合类结合 将Arrays.asList返回的集合作为ArrayList的构造参数
ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));

最后

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

那么这个知识点,你get到了吗?get到了,那来继续关注我。没get到?来来来,咱俩单独聊聊。

原文链接:https://blog.csdn.net/adparking/article/details/117322970



所属网站分类: 技术文章 > 博客

作者:我是小豆丁

链接:http://www.javaheidong.com/blog/article/207605/f6007c814df6c34dc102/

来源:java黑洞网

任何形式的转载都请注明出处,如有侵权 一经发现 必将追究其法律责任

5 0
收藏该文
已收藏

评论内容:(最多支持255个字符)