Arrays.asList 使用细节

通常初始化后使用如下,但是报错 UnsupportOperationException....

根据提示信息,就是调用add()方法时抛出了异常。顺着堆栈信息往上找,提示的是AbstractList类的108行出了异常,这一行所在方法的具体实现如下:

//108行
public boolean add(E var1) {
        this.add(this.size(), var1);
        return true;
 }
//148行
public void add(int var1, E var2) {
        throw new UnsupportedOperationException();
 }

//下面看下Arrays.asList的具体实现
@SafeVarargs
public static <T> List<T> asList(T... var0) {
  return new Arrays.ArrayList(var0);
}

//Arrays.ArrayList 实现
private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, Serializable {
    private static final long serialVersionUID = -2764017481108945198L;
    private final E[] a;
    //关键
    ArrayList(E[] var1) {
        this.a = (Object[])Objects.requireNonNull(var1);
    }

    public int size() {
        return this.a.length;
    }

    public Object[] toArray() {
        return (Object[])this.a.clone();
    }

    public <T> T[] toArray(T[] var1) {
        int var2 = this.size();
        if (var1.length < var2) {
            return Arrays.copyOf(this.a, var2, var1.getClass());
        } else {
            System.arraycopy(this.a, 0, var1, 0, var2);
            if (var1.length > var2) {
                var1[var2] = null;
            }

            return var1;
        }
    }

    public E get(int var1) {
        return this.a[var1];
    }

    public E set(int var1, E var2) {
        Object var3 = this.a[var1];
        this.a[var1] = var2;
        return var3;
    }

    public int indexOf(Object var1) {
        Object[] var2 = this.a;
        int var3;
        if (var1 == null) {
            for(var3 = 0; var3 < var2.length; ++var3) {
                if (var2[var3] == null) {
                    return var3;
                }
            }
        } else {
            for(var3 = 0; var3 < var2.length; ++var3) {
                if (var1.equals(var2[var3])) {
                    return var3;
                }
            }
        }

        return -1;
    }

    public boolean contains(Object var1) {
        return this.indexOf(var1) != -1;
    }

    public Spliterator<E> spliterator() {
        return Spliterators.spliterator(this.a, 16);
    }

    public void forEach(Consumer<? super E> var1) {
        Objects.requireNonNull(var1);
        Object[] var2 = this.a;
        int var3 = var2.length;

        for(int var4 = 0; var4 < var3; ++var4) {
            Object var5 = var2[var4];
            var1.accept(var5);
        }

    }

    public void replaceAll(UnaryOperator<E> var1) {
        Objects.requireNonNull(var1);
        Object[] var2 = this.a;

        for(int var3 = 0; var3 < var2.length; ++var3) {
            var2[var3] = var1.apply(var2[var3]);
        }

    }

    public void sort(Comparator<? super E> var1) {
        Arrays.sort(this.a, var1);
    }
}
//此处可以发现,改内部类继承了 AbstractList 但是却没有实现其的add,remove等方法。(下面给出AbstractList 的定义)
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
    protected transient int modCount = 0;

    protected AbstractList() {
    }

    public boolean add(E var1) {
        this.add(this.size(), var1);
        return true;
    }

    public abstract E get(int var1);

    public E set(int var1, E var2) {
        throw new UnsupportedOperationException();
    }

    public void add(int var1, E var2) {
        throw new UnsupportedOperationException();
    }

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

最后,我们发现,此ArrayList不是彼ArrayList。这个ArrayList是Arrays工具类中实现的内部静态类,我们可以发现,这个类集成了AbstractList类,但是并没有重写add()方法,所以在我们的示例代码标记(2)处调用add()方法时,实际是调用父类AbstractList的add()方法,这也就回到了开头分析的那两个add()方法了,它们都没有具体实现,只会抛出UnsupportedOperationException。

结论总结:

  我们调用Arrays的asList()方法将数组转换成List时返回的是Arrays的静态内部类ArrayList,它自身并未重写add()方法,而其父类AbstractList实现的add()方法只会抛出UnsupportedOperationException,导致我们调用Arrays的静态内部类ArrayList的add()方法时,实际调用的是只会抛出UnsupportedOperationException的AbstractList的add()方法,这就是异常出现的原因了。

解决办法

4、 解决方案 
可以自己写个工具类转换方法,可以参考一下代码:

private static <E> List<E> transferArrayList(E[] array){
List<E> transferedList = new ArrayList<>();
Arrays.stream(array).forEach(arr -> transferedList.add(arr));
return transferedList;
}

原文地址:https://www.cnblogs.com/weixiaotao/p/11345125.html

时间: 2024-08-07 15:16:51

Arrays.asList 使用细节的相关文章

【Java基础】一个有意思的泛型方法Arrays.asList(T... a)

总结 利用Arrays.asList方法返回的List是不允许add和remove的,这种list的长度不可变,因为底层依然是写数组. Arrays.asList的返回值是调用是传入T类型的List,所以传入啥,返回啥的列表 T... a 底层本来就是转换为T[] x的数组,所以如果传入的T是数组,最后的底层参数是二维数组T[][] y. Arrays.asList(T... a)方法的作用 将数组转为集合的方法,返回的是List集合.和Collection的toArray对应,是数组和集合间相

浅谈Arrays.asList()方法的使用

首先,该方法是将数组转化为list.有以下几点需要注意: (1)该方法不适用于基本数据类型(byte,short,int,long,float,double,boolean) (2)该方法将数组与列表链接起来,当更新其中之一时,另一个自动更新 (3)不支持add和remove方法 上代码: 1 package com.hdu.test; 2 3 import java.util.Arrays; 4 import java.util.List; 5 6 abstract public class

Arrays.asList 小结

Arrays.asList()返回一个实现了List接口的类,但是不支持add,remove等改变尺寸的操作. Arrays.asList源码如下 1 public static <T> List<T> asList(T... a) { 2 return new ArrayList<>(a); 3 } 这里有一个大坑,我开始没仔细看,以为返回的就是java.util.ArrayList,可实际上此ArrayList是Arrays里面的私有静态内部类,代码如下所示. 1

Arrays.asList () 不可添加或删除元素的原因

Java中奖数组转换为List<T>容器有一个很方便的方法 Arrays.asList(T ... a),我通过此方法给容器进行了赋值操作,接着对其进行 添加元素,却发现会抛出一个(java.lang.UnsupportedOperationException)异常. 看了一下源码: public static <T> List<T> asList(T... a) { return new ArrayList<T>(a); } 定位到 ArrayList 内

Arrays.asList的哪点事

import java.util.ArrayList; import java.util.Arrays; import java.util.HashSet; import java.util.List; public class ArraysAsListDemo { public static void main(String[] args) { List<String> source1 = new ArrayList<String>(Arrays.asList("1&q

Arrays.asList的用法:

Arrays.asList 返回数组而非List. List的一个典型的特性就是其长度是可变的,我们可以很方便地对它进行插入和删除元素的操作,这是它与数组所存在的一个很大的区别,后者的长度是固定的,而且我们不能从数组中删除元素,只能修改元素的值. 利用Arrays.asList(array)返回一个List,然而这个返回的是只读的List不支持add和remove的操作. JDK文档是这么说的: public static <T> List<T> asList(T... a) 返回

Arrays.asList()方法

// Collection<Integer> integers = new ArrayList<Integer>();//(Arrays.asList(1,2,3,4,5)); Collection<Integer> integers = new ArrayList<Integer>(Arrays.asList(1,2,3,4,5));// integers = Arrays.asList(1,2,3,4,5); System.out.println(int

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

在平时的开发过程中,我们知道可以将一个Array的对象转化为List.这样的操作,我们只要采用Arrays.asList这个方法就行了.笔者前段时间一直用这个方法,有一天,我发现通过Arrays.asList得到的List无法进行add和remove等操作. 下面是一段很简单的测试代码: public class MainFacade { public static void main(String[] args) { List<Integer> list = Arrays.asList(1,

Arrays.asList的用法

使用工具类Arrays.asList()把数组转换成集合时,不能使用其修改集合相关的方法,它的add/remove/clear方法会抛出UnsupportOperationException异常说明:asList的返回对象是一个Arrays内部类,并没有实现集合的修改方法.Arrays.asList体现的是适配器模式,只是转换接口,后台的数据仍是数组.String[] str = new String[]{"1","2"};List list = Arrays.as