Arrays.asList的用法

本文深入解析了Java中Arrays.asList方法的内部实现,揭示了其返回的List实际上是对数组的只读包装,强调了不能进行add或remove操作的特点。并详细解释了在不同JDK版本下,该方法对基本类型数组的支持差异。

Arrays.asList 返回数组而非List。
List的一个典型的特性就是其长度是可变的,我们可以很方便地对它进行插入和删除元素的操作,这是它与数组所存在的一个很大的区别,后者的长度是固定的,而且我们不能从数组中删除元素,只能修改元素的值。

利用Arrays.asList(array)返回一个List,然而这个返回的是只读的List不支持add和remove的操作。

JDK文档是这么说的:
public static List asList(T… a) 返回一个受指定数组支持的固定大小的列表。(对返回列表的更改会“直接写”到数组。)此方法同 Collection.toArray() 一起,充当了基于数组的 API 与基于 collection 的 API 之间的桥梁。返回的列表是可序列化的,并且实现了 RandomAccess。此方法还提供了一个创建固定长度的列表的便捷方法,该列表被初始化为包含多个元素: List stooges = Arrays.asList(“Larry”, “Moe”, “Curly”);

可以看出这是对数组的一个简单包装,提供了一个可按照List方式访问的外壳而已,其本质还是数组。

如果用Arrays.asList(arrays)得到的List往里面插入数据时会报java.lang.UnsupportedOperationException 异常

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

2、注意:

JDK 1.4对java.util.Arrays.asList的定义,函数参数是Object[]。所以,在1.4中asList()并不支持基本类型的数组作参数。
JDK 1.5中,java.util.Arrays.asList的定义,函数参数是Varargs, 采用了泛型实现。同时由于autoboxing的支持,使得可以支持对象数组以及基本类型数组。
  不过在使用时,当传入基本数据类型的数组时,会出现小问题,会把传入的数组整个当作返回的List中的第一个元素,例如:

public static void main(String[] args){
     int[] a1 = new int[]{1,2,3};
     String[] a2  = new String[]{"a","b","c"};
     System.out.println(Arrays.asList(a1));
   System.out.println(Arrays.asList(a2));
 }

打印结果如下:
  [[I@dc8569]
  [a, b, c]

3、Arrays.asList源码:

   public static <T> List<T> asList(T... a) {
       return new ArrayList<T>(a);
    }

这里的ArrayList并不是java.util.ArrayList,而是Arrays的内部类:

 
private static class ArrayList<E> extends AbstractList<E>implements  RandomAccess,java.io.Serializable {
 
		private static final long serialVersionUID = -2764017481108945198L;
		private final E[] a;
	
		ArrayList(E[] array) {
	    if (array == null)
	      throw new NullPointerException();
	    	a = array;
	    }
 
	     public int size() {
             return a.length;
         }
 
	     public Object[] toArray() {
             return a.clone();
         }
 
	     public <T> T[] toArray(T[] a) {
             int size = size();
             if (a.length < size)
                 return Arrays.copyOf(this.a, size, (Class<? extends T[]>) a.getClass());
             System.arraycopy(this.a, 0, a, 0, size);
            if (a.length > size)
                 a[size] = null;
             return a;
         }
 
	         public E get(int index) {
             return a[index];
         }
 
	      public E set(int index, E element) {
            E oldValue = a[index];
             a[index] = element;
            return oldValue;
         }
 
	      public int indexOf(Object o) {
           if (o == null) {
               for (int i = 0; i < a.length; i++)
                    if (a[i] == null)
                        return i;
           } else {
              for (int i = 0; i < a.length; i++)
                  if (o.equals(a[i]))
                        return i;
             }
             return -1;
         }
 
	    public boolean contains(Object o) {
           return indexOf(o) != -1;
        }    
}

我们可以看到该内部类继承的是AbstractList,下面是AbstractList的add和remove方法源码:

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

所以,当我们对Arrays.asList返回的List进行添加或删除时将会报 java.lang.UnsupportedOperationException 异常。

### 用法说明 `Arrays.asList()` 是 Java 提供的一个静态方法,主要用于将数组转换为 `List` 集合。其定义如下: ```java public static <T> List<T> asList(T... a) ``` 该方法返回的 `List` 是 `java.util.Arrays` 类中的一个内部类 `ArrayList` 的实例,与 `java.util.ArrayList` 并不相同。这个生成的 `List` 是基于固定大小的数组,因此它不支持对集合大小进行修改的操作,例如添加、删除等操作,否则会抛出 `UnsupportedOperationException` 异常 [^3]。 ### 示例 以下是一个简单的示例,展示如何使用 `Arrays.asList()` 方法: ```java import java.util.Arrays; import java.util.List; public class Main { public static void main(String[] args) { String[] array = {"Apple", "Banana", "Cherry"}; List<String> list = Arrays.asList(array); System.out.println(list); } } ``` 上述代码将数组 `array` 转换为一个 `List`,并输出结果: ``` [Apple, Banana, Cherry] ``` 需要注意的是,由于返回的 `List` 是基于原始数组的,因此对数组的修改会影响到 `List`,反之亦然 [^2]。 ### 注意事项 1. **不可变大小**:通过 `Arrays.asList()` 创建的 `List` 是固定大小的,不能添加或删除元素。尝试调用 `add()` 或 `remove()` 方法会导致 `UnsupportedOperationException` 异常 。 2. **与 `java.util.ArrayList` 的区别**:虽然返回的类型是 `List`,但其内部实现是 `Arrays` 类的内部类 `ArrayList`,而不是 `java.util.ArrayList`。因此,某些方法的行为可能与预期不同 [^3]。 3. **引用关系**:创建的 `List` 与原始数组共享相同的引用。因此,对数组的修改会反映在 `List` 中,反之亦然 [^2]。 ### 适用场景 `Arrays.asList()` 方法适用于需要将数组快速转换为 `List` 的场景,特别是在初始化集合时非常方便。例如,在需要传递一个 `List` 参数时,可以直接使用数组进行转换,而不必手动创建 `List` 并逐个添加元素 [^2]。 此外,该方法也常用于遍历数组,或者在不需要修改集合大小的情况下进行集合操作 [^4]。 ### 相关问题 1. `Arrays.asList()` 和 `new ArrayList<>(Arrays.asList())` 有什么区别? 2. 如何避免 `Arrays.asList()` 返回的 `List` 无法修改的问题? 3. `Arrays.asList()` 是否支持基本数据类型数组的转换?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值