Arrays.asList() 返回的list不能add,remove

一.Arrays.asList() 返回的list不能add,remove

Arrays.asList()返回的是List,而且是一个定长的List,所以不能转换为ArrayList,只能转换为AbstractList

原因在于asList()方法返回的是某个数组的列表形式,返回的列表只是数组的另一个视图,而数组本身并没有消失,对列表的任何操作最终都反映在数组上. 所以不支持remove,add方法的

下面是一段很简单的测试代码:

java 复制代码
public class MainFacade {  
    public static void main(String[] args) {  
        List<Integer> list = Arrays.asList(1,2,3);  
        list.add(5);  
        System.out.print(list.toString());  
    }  
}  

不过上面的代码会throw出一个UnsupportedOperationException这样的异常

Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.add(AbstractList.java:148) at java.util.AbstractList.add(AbstractList.java:108) at org.popkit.MainFacade.main(MainFacade.java:14) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:606) at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

终其原因是Arrays.asList方法返回的ArrayList是继承自AbstractList同时实现

了RandomAccess和Serializable接口,定义如下:

复制代码
private static class ArrayList<E> extends AbstractList<E>  implements RandomAccess, java.io.Serializable  

我们再来看看AbstractList这个类的定义:

java 复制代码
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E>   

这时我们发现AbstractList这个类的set add remove方法定义如下:

java 复制代码
public void add(int index, E element) {  
    throw new UnsupportedOperationException();  
}  
  
public E set(int index, E element) {  
    throw new UnsupportedOperationException();  
}  
  
public E remove(int index) {  
    throw new UnsupportedOperationException();  
}  

现在知道了它throw UnsupportedOperationException异常的原因了。

通过上面的分析,我们知道,其实通过asList方法得到的List是只读的,那么平时我们怎样避免这样的错误发生?我们可以采用如下方法:

java 复制代码
List<Integer> list = new ArrayList<>(Arrays.asList(1,2,3));  

二、Arrays.asList()陷阱

代码如下:

Java代码

java 复制代码
public static void main(String[] args) {  
        int[] data = {1,2,3,4,5};  
        List list = Arrays.asList(data);  
        System.out.println("列表中的元素数量是:" + list.size());  
    }  

注意这里输出的数量是1,原因是,在Arrays.asList中,其接收的参数原型其实是泛型变长参数来的,而基本类型是不能作为范型的参数,按道理应该使用包装类型,但这里缺没有报错,

因为数组是可以泛型化的,所以转换后在list中就有一个类型为int的数组

Java代码

java 复制代码
        int[] data = {1,2,3,4,5};  
        List list = Arrays.asList(data);  
        System.out.println("元素类型:" + list.get(0).getClass());  
        System.out.println("前后是否相等:"+data.equals(list.get(0)));  

可以看到,输出的为元素类型:class [I

因为jvm不可能输出array类型,array类型属于java.lang.reflect包,通过反射访问

数组的这个类,编译时候生成的。所以要改为:

Java代码

java 复制代码
Integer[] data = {1,2,3,4,5};  
List list = Arrays.asList(data);  
System.out.println("列表中的元素数量是:" + list.size());  

此外,假如如下代码:

Java代码

java 复制代码
//枚举,声明一个星期  
enum Week{Sun,Mon, Tue, Wed,Thu,Fri,Sat}      
public static void main(String[] args) {  
        //工作日  
        Week[] workDays = {Week.Mon, Week.Tue, Week.Wed,Week.Thu,Week.Fri};  
        //转换为列表  
        List<Week> list = Arrays.asList(workDays);  
        //增加周六也为工作日  
        list.add(Week.Sat);  

注意这里用add会出错,因为是arrays.aslist中,看代码可以看到这里返回的

ArrayList不是原来的传统意义上的java.util.arraylist了,而是自己工具类的一个

静态私有内部类,并没有提供add方法,要自己实现,所以这里是出错了,因此,

除非确信array.aslist后长度不会增加,否则谨慎使用:

List<String> abc=Arrays.asList("a","b","c"),因为这样的长度是无法再add的了

相关推荐
微露清风2 分钟前
系统性学习C++-第十讲-stack 和 quene
java·c++·学习
抠脚学代码3 分钟前
Linux开发-->驱动开发-->字符设备驱动框架
linux·数据结构·驱动开发
一蓑烟雨任平生√7 分钟前
两种上传图片的方式——91张先生
java·ossinsight
凤凰战士芭比Q21 分钟前
部署我的世界-java版服务器-frp内网穿透
java·服务器
小肖爱笑不爱笑22 分钟前
2025/11/5 IO流(字节流、字符流、字节缓冲流、字符缓冲流) 计算机存储规则(ASCII、GBK、Unicode)
java·开发语言·算法
CodeCraft Studio33 分钟前
PPT处理控件Aspose.Slides教程:使用Java将PowerPoint笔记导出为PDF
java·笔记·pdf·powerpoint·aspose·ppt转pdf·java将ppt导出pdf
手握风云-34 分钟前
Java 数据结构第二十八期:反射、枚举以及 lambda 表达式
java·开发语言
ᐇ95939 分钟前
Java Vector集合全面解析:线程安全的动态数组
java·开发语言
毕设源码-朱学姐1 小时前
【开题答辩全过程】以 广州网红点打卡介绍网站为例,包含答辩的问题和答案
java·eclipse
程序定小飞1 小时前
基于springboot的web的音乐网站开发与设计
java·前端·数据库·vue.js·spring boot·后端·spring