php中文网 | cnphp.com

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 403|回复: 0

Arrays.asList()你真的知道怎么用吗?

[复制链接]

3142

主题

3152

帖子

1万

积分

管理员

Rank: 9Rank: 9Rank: 9

UID
1
威望
0
积分
7956
贡献
0
注册时间
2021-4-14
最后登录
2024-11-22
在线时间
763 小时
QQ
发表于 2022-9-6 08:37:16 | 显示全部楼层 |阅读模式
发现问题
前几天在看别人的项目的时候,发现一个问题,简单复现一下这个问题
// 注意这是一个Integer对象的数组哦
Integer[] arr = new Integer[]{9999,88,77};
List<Integer> list = Arrays.asList(arr);
// 执行以下操作,有问题么?
list.add(1);
list.remove(0);

好的,如果你觉得没错,和我刚开始的想法一致。在我没有认真学习这个asList方法时,天真的以为没有问题,顾名思义啊,就是把数组转换成List呗。


// 恭喜,喜提报错,如果是这样测试该找你麻烦了,/(ㄒoㄒ)/~~
Exception in thread "main" java.lang.UnsupportedOperationException
        at java.util.AbstractList.add(AbstractList.java:148)
        at java.util.AbstractList.add(AbstractList.java:108)
        at TestForAsList.main(TestForAsList.java:13)

废话不多说,我们直接进源码里面看看。
@SafeVarargs
@SuppressWarnings("varargs")
public static <T> List<T> asList(T... a) {
    return new ArrayList<>(a);  // 看起来没有任何异常,emm。然而呢,这个ArrayList并不是Java.util包下的ArrayList,而是一个在Arrays下实现的内部类。
}
image.png
我们发现这个静态内部类里面并没有实现List的add和remove方法。那么子类将延用父类AbstractList的方法实现,这个继承应没有什么问题。
// 我们进入这个父类的实现,发现了报错的根源。。
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
   
        public void add(int index, E element) {
        throw new UnsupportedOperationException();
    }


    public E remove(int index) {
        throw new UnsupportedOperationException();
    }
}

总结:

究其原因,就是这个asList方法返回的是一个内部类,只实现了一些遍历以及更新的方法。下次使用它的时候需要注意一下。

扩展知识点
数组值的变化
然后我就发现了这个很有意思的点,就是这个Arrays包下的ArrayList用的是构造器传进来的数组,并不像我们原来认为的ArrayList的那样会拷贝数组然后创建一个新的数组。

这意味着我们在对这个List进行set的改动时,我们同时会更改原数组的值。

public static void main(String[] args) {
        Integer[] arr = new Integer[]{9999,88,77};
        List<Integer> list = Arrays.asList(arr);
        
        list.set(1,0);
        System.out.println(list.get(1));  // 输出0
        System.out.println(arr[1]);  // 同样也输出了0
}
int[]数组
我们都知道int是基本数据类型,如果我们向asList()方法里面传入一个int[]数组会发生什么呢?

我们知道泛型的话需要的是对象类型,基础数据类型是不能作为泛型的。
// 此时int[]会作为一个对象类型,然后转换为list。此时长度为1,且可以正常的取出来作为list元素
public static void main(String[] args) {
        int[] arr = new int[]{9999,88,77};
        List<int[]> list = Arrays.asList(arr);
        System.out.println(list.size());
        System.out.println(list.get(0)[0]);
        System.out.println(list.get(0)[1]);
        System.out.println(list.get(0)[2]);
}






上一篇:【题解笔记】PTA基础6-10:阶乘计算升级版
下一篇:EntityFrameworkCore 模型自动更新(上)
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|php中文网 | cnphp.com ( 赣ICP备2021002321号-2 )

GMT+8, 2024-11-22 20:48 , Processed in 0.981891 second(s), 45 queries , Gzip On.

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2020, Tencent Cloud.

申明:本站所有资源皆搜集自网络,相关版权归版权持有人所有,如有侵权,请电邮(fiorkn@foxmail.com)告之,本站会尽快删除。

快速回复 返回顶部 返回列表