为什么Java里的Arrays.asList不能用add和remove方法?

本文解析了使用Arrays.asList方法创建List实例时遇到的不可修改问题,并提供了正确的转换方式。

在平时的开发过程中,我们知道可以将一个Array的对象转化为List。这样的操作,我们只要采用Arrays.asList这个方法就行了。笔者前段时间一直用这个方法,有一天,我发现通过Arrays.asList得到的List无法进行add和remove等操作。

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

 


 
  1. public class MainFacade {

  2. public static void main(String[] args) {

  3. List<Integer> list = Arrays.asList(1,2,3);

  4. list.add(5);

  5. System.out.print(list.toString());

  6. }

  7. }

 

 

不过上面的代码会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接口,定义如下:  

 

 


 
  1. private static class ArrayList<E> extends AbstractList<E>

  2. implements RandomAccess, java.io.Serializable

  3.  


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

 

    public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> 

 

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

 


 
  1. public void add(int index, E element) {

  2. throw new UnsupportedOperationException();

  3. }

  4.  
  5. public E set(int index, E element) {

  6. throw new UnsupportedOperationException();

  7. }

  8.  
  9. public E remove(int index) {

  10. throw new UnsupportedOperationException();

  11. }

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

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

 

    List<Integer> list = new ArrayList<>(Arrays.asList(1,2,3));


 

### Arrays.asList() 返回的列表不可修改的原因 `Arrays.asList()` 方法返回的列表实际上是 `java.util.Arrays` 类中的一个内部类 `ArrayList` 的实例,而不是我们通常使用的 `java.util.ArrayList`。这个内部类 `ArrayList` 继承自 `AbstractList`,并且没有实现 `add()` `remove()` 方法[^1]。因此,当尝试对通过 `Arrays.asList()` 创建的列表调用 `add()` 方法时,会抛出 `UnsupportedOperationException` 异常。 此外,`Arrays.asList()` 返回的列表是一个固定大小的视图,其底层数据结构仍然是传入的数组。任何对列表的操作都会直接影响到原始数组,但无法改变列表的大小[^4]。 --- ### 解决方案 为了使列表能够动态调整大小并支持 `add()` `remove()` 操作,可以将 `Arrays.asList()` 返回的列表包装到一个真正的 `java.util.ArrayList` 实例中。以下是具体的实现方式: ```java import java.util.ArrayList; import java.util.Arrays; import java.util.List; public class Main { public static void main(String[] args) { String[] array = {"apple", "banana", "cherry"}; // 使用 Arrays.asList() 返回固定大小的列表 List<String> fixedSizeList = Arrays.asList(array); // 将固定大小的列表转换为可变大小的 ArrayList List<String> mutableList = new ArrayList<>(fixedSizeList); // 现在可以使用 add() 方法 mutableList.add("orange"); System.out.println(mutableList); // 输出: [apple, banana, cherry, orange] } } ``` 在这个例子中,`Arrays.asList(array)` 返回的是一个固定大小的列表,而通过 `new ArrayList<>(fixedSizeList)` 创建了一个新的、可变大小的 `ArrayList`,从而支持了 `add()` `remove()` 操作[^3]。 --- ### 注意事项 1. 如果需要对基本数据类型(如 `int[]`, `double[]`)使用类似的功能,则需要先将其转换为对应的包装类数组(如 `Integer[]`, `Double[]`),因为 `Arrays.asList()` 不支持直接处理基本数据类型数组[^4]。 2. 在某些情况下,如果只需要读取操作而不涉及修改列表大小,可以直接使用 `Arrays.asList()` 返回的列表,以避免额外的内存开销。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值