文档章节

Arrays.asList回来的List与new ArrayList的区别

moreThink
 moreThink
发布于 2016/04/08 17:05
字数 805
阅读 212
收藏 8

精选30+云产品,助力企业轻松上云!>>>

      转自: http://www.myexception.cn/program/1351095.html 

      前几天写代码的时候用到将Set转换为List然后继续进行操作,向里面添加元素的时候报错了,代码逻辑类似下面:

import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

public class Test {

	public static void main(String[] args) {
		
		Set<String> set = new HashSet<String>();
		set.add("a");
		set.add("b");
		set.add("c");
		
		//某些判断逻辑
		
		List<String> list = Arrays.asList(set.toArray(new String[0]));
		list.add("d");
		
		System.out.println(list);
	}
}

 代码执行到list.add("d");这一行的时候报错了,报错如下图:



    难道Arrays.asList返回的List与new出来的有什么不同呢!根据报错的信息,是AbstractLst不支持add操作,从源码的结构看List的很多实现类都继承了AbstractList,包括常用的ArrayList。那么
Arrays.asList是怎么实现的呢,顺着源码追踪看下去,它的实现非常简单,如下:

/**
     * 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 a the array by which the list will be backed
     * @return a list view of the specified array
     */
    @SafeVarargs
    public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }

    注释中倒是说明了返回的是个固定大小的List,确是用ArrayList实现的,当liuyh17211在追踪进ArrayList的时候真相大白了,原来这个ArrayList不是util包中的ArrayList,而只是Arrays类的一个继承了AbstractList内部类,所有代码如下:

虽然在编辑器上的引用显示的是util包中的,实际在运行的时候会先找到内部类。

/**
     * @serial include
     */
    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中,明确提到了不覆盖就会抛UnsupportedOperationException异常的方法有3个:add(int index, E element),set(int index, E element),remove(int index)。而上面的代码中只覆盖了set方法,可能会调用这几个方法的add(E element),clear(),addAll(int index, Collection<? extends E> c),甚至iterator()方法都没有覆盖,也就是说上面的几个方法都可能在调用中报错,liuyh17211试了一下,只要不去修改list中的值,调用iterator()方法是没问题的。

    由此可见JDK设计的这个返回List,只支持遍历和取值,不能做任何修改,只能作为传递值的桥梁。

 

注:我的jdk版本是1.7.0_09,源码不一定和其他版本的jdk下一致。

moreThink
粉丝 2
博文 42
码字总数 16535
作品 0
浦东
高级程序员
私信 提问
加载中
请先登录后再评论。
摘要

http://www.programcreek.com/2013/09/top-9-questions-for-java-map/ http://www.programcreek.com/2013/03/hashmap-vs-treemap-vs-hashtable-vs-linkedhashmap/ http://www.programcreek.c......

完美世界
2014/06/01
41
0
使用Arrays.asList()时遇到的问题

Arrays.asList()是JDK自带的一个将数组转换为集合的方法, 但是在使用中发现, 使用该方式在将数组转换为集合之后,相应的集合修改方法使用时会抛出异常, 如果add/remove/clear等.打开JDK源码,...

osc_9wny6wso
2019/09/16
2
0
总结软件开发过程中最常犯的10个问题

先总结一下Top10吧 Array转ArrayList 判断一个数组是否包含某个值 在循环内部删除List中的一个元素 HashTable与HashMap 使用集合原始类型(raw type) 访问级别 ArrayList和LinkedList 可变与...

爱码仕i
2019/07/08
4
0
Java开发最常犯的10个错误,打死都不要犯!

阅读目录 这个列表总结了10个Java开发人员最常犯的错误。 1、Array转ArrayList 当需要把Array转成ArrayList的时候,开发人员经常这样做: List list = Arrays.asList(arr); Arrays.asList()会...

world_snow
03/31
0
0
List、Set、数组之间的转换

List、Set、数组之间的转换 数组转Collection 使用Apache Jakarta Commons Collections: CollectionUtils.addAll()方法的实现很简单,只是循环使用了Collection的add()方法而已。 如果只是想...

maskleo
2015/04/20
9
0

没有更多内容

加载失败,请刷新页面

加载更多

4.17. HTTP 请求走私

文章目录 4.17. HTTP 请求走私 4.17.1. 简介 4.17.2. 成因 4.17.3. 分类 4.17.4. 攻击 4.17.4.1. CL不为0的GET请求 4.17.4.2. CL-CL 4.17.4.3. CL-TE 4.17.4.4. TE-CL 4.17.4.5. TE-TE 4.17......

osc_2wq8ft8d
19分钟前
11
0
2020年蓝桥杯B组个人题解(热的,不知道对错)

文章目录 A B C D E F G H I J 总结 现在是蓝桥杯刚结束,趁着有记忆,写下这篇博客,不知道对错,如果我错了,请指出。 A 因为是到0就结束了,那么每次看看-600是否结束,如果没有结束就+30...

osc_1m04dwae
20分钟前
0
0
AXI4_full协议详解

AXI4-full协议介绍 AXI4.0-full包含突发控制信号,所以可以进行突发传输,在只指定一次地址后,可以一次传输多达256个数据(数据的宽度取决于带宽)。主要用于往DDR或者OCM中写入大量数据时使用...

whoisliang
20分钟前
8
0
【VMWare虚拟机Centos7安装配置redis及Redis常用命令合集】

前置依赖 1.Redis依赖gcc和tcl环境,所以要先安装gcc和tcl 执行下面语句安装gcc: yum install gcc-c++ 若提示是否继续,输入y继续安装 执行下面语句安装tcl: yum install -y tcl 下载red...

osc_s4durjz5
20分钟前
3
0
LeetCode 323. 无向图中连通分量的数目(并查集)

文章目录 1. 题目 2. 解题 1. 题目 给定编号从 0 到 n-1 的 n 个节点和一个无向边列表(每条边都是一对节点),请编写一个函数来计算无向图中连通分量的数目。 示例 1:输入: n = 5 和 edge...

osc_lk0wespa
22分钟前
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部