采用Arrays.asList后对list进行add操作报异常:UnsupportedOperationException

本文深入探讨了Java中Arrays.asList方法生成的List与常规ArrayList之间的关键差异,特别是add方法的实现方式。通过代码示例说明了如何避免UnsupportedOperationException异常,并提供了解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

大清早的来了,把周五运行的代码一波,

出现异常

操作:把一个用逗号分隔字符串split成一个数组,我想转成list结果就来了一波Arrays.asList,而后又想对生成的list做添加,采用了add操作,结果就抛异常了----

结合网上查阅加原来看

1、Arrays.asList生成的list是否和我们平时使用的new ArrayList是否一致

   /**
     * Returns a fixed-size list backed by the specified array.  (Changes to
     * the returned list "write through" to the array.)  This method acts
     * as bridge between array-based and collection-based APIs, in
     * combination with {@link Collection#toArray}.  The returned list is
     * serializable and implements {@link RandomAccess}.
     *
     * <p>This method also provides a convenient way to create a fixed-size
     * list initialized to contain several elements:
     * <pre>
     *     List&lt;String&gt; stooges = Arrays.asList("Larry", "Moe", "Curly");
     * </pre>
     *
     * @param <T> the class of the objects in the array
     * @param a the array by which the list will be backed
     * @return a list view of the specified array
     */
    @SafeVarargs
    @SuppressWarnings("varargs")
    public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }

Arrays的源码中也是返回的个new ArrayList,这就有点蒙蔽了,ctrl按住点ArrayList 结果是个Arrays里面内部类

返回的是这个内部类的ArrayList,但是这个继承自AbstractList这个抽象类,在AbstractList没有对add方法进行重写,而这个内部类的ArrayList也没有重写add方法,调用内部类的ArrayList的add方法相当于调用了父类AbstractList的add方法,源码一看,这直接就抛出了UnsupportedOperationException异常;

public boolean add(E e) {
    add(size(), e);
    return true;
}
public void add(int index, E element) {
    throw new UnsupportedOperationException();
}

这就得再看看我们的java util里面的ArrayList了,也是继承自AbstractList,但是它重写了add方法

public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}

这样就不会调用到父类的直接抛出异常的add方法,无奈,只好把内部类的arrayList再转一遍到util中的arrayList,调用传参为集合的构造方法进行转换;

public ArrayList(Collection<? extends E> c) {
    elementData = c.toArray();
    if ((size = elementData.length) != 0) {
        // c.toArray might (incorrectly) not return Object[] (see 6260652)
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    } else {
        // replace with empty array.
        this.elementData = EMPTY_ELEMENTDATA;
    }
}
String str = "a,b,c,d";
String[] strArr = str.split(",");
List<String> list = Arrays.asList(strArr);
list.add("e");
//上面的代码会抛出异常  执行下面时,把上面一行的代码注释掉  //list.add("e");
List<String> utilList = new ArrayList(list);
utilList .add("e");
System.out.println(utilList.toString());

//运行结果
//[a, b, c, d, e]

Arrays.asList返回的arrayList根本就没有对add和remove方法重写,还是自己太菜,没踩过的坑,记录一下。

### 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、付费专栏及课程。

余额充值