Arrays.asList()vs Collections.singletonList()

Collections.singletonList(something)不可变的,

Collections.singletonList(something)返回的列表所做的任何更改将导致UnsupportedOperationException 。

Arrays.asList(something)允许Arrays.asList(something) 更改  。

此外,由Collections.singletonList(something)返回的List的容量将始终为1,

Arrays.asList(something)的容量将为已支持数组的大小。

/**

     * Returns an immutable list containing only the specified object.
     * The returned list is serializable.
     *
     * @param  <T> the class of the objects in the list
     * @param o the sole object to be stored in the returned list.
     * @return an immutable list containing only the specified object.
     * @since 1.3
     */
    public static <T> List<T> singletonList(T o) {
        return new SingletonList<>(o);
    }

    /**
     * @serial include
     */
    private static class SingletonList<E>
        extends AbstractList<E>
        implements RandomAccess, Serializable {

        private static final long serialVersionUID = 3093736618740652951L;

        private final E element;

        SingletonList(E obj)                {element = obj;}

        public Iterator<E> iterator() {
            return singletonIterator(element);
        }

        public int size()                   {return 1;}

        public boolean contains(Object obj) {return eq(obj, element);}

        public E get(int index) {
            if (index != 0)
              throw new IndexOutOfBoundsException("Index: "+index+", Size: 1");
            return element;
        }

        // Override default methods for Collection
        @Override
        public void forEach(Consumer<? super E> action) {
            action.accept(element);
        }
        @Override
        public boolean removeIf(Predicate<? super E> filter) {
            throw new UnsupportedOperationException();
        }
        @Override
        public void replaceAll(UnaryOperator<E> operator) {
            throw new UnsupportedOperationException();
        }
        @Override
        public void sort(Comparator<? super E> c) {
        }
        @Override
        public Spliterator<E> spliterator() {
            return singletonSpliterator(element);
        }
    }

附录:

package com.ysyc.invoicecertify.util.mockservice;

import java.util.Arrays;
import java.util.List;

/**
 *
 * 本类演示了Arrays类中的asList方法
 * 通过四个段落来演示,体现出了该方法的相关特性.
 *
 * (1) 该方法对于基本数据类型的数组支持并不好,当数组是基本数据类型时不建议使用
 * (2) 当使用asList()方法时,数组就和列表链接在一起了.
 *     当更新其中之一时,另一个将自动获得更新。
 *     注意:仅仅针对对象数组类型,基本数据类型数组不具备该特性
 * (3) asList得到的数组是的没有add和remove方法的
 *
 * 通过查看Arrays类的源码可以知道,asList返回的List是Array中的实现的
 * 内部类,而该类并没有定义add和remove方法.另外,为什么修改其中一个,另一个也自动
 * 获得更新了,因为asList获得List实际引用的就是数组
 */
public class AsListTest {

    public static void main(String[] args) {

        /* 段落一:基本数据类型使用asList中的问题 */

        /* 说明:虽然在JDK1.6中能够将基本数据类型的数组转换成List,但还是有个缺陷 */
        int[] a_int = { 1, 2, 3, 4 };
        /* 预期输出应该是1,2,3,4,但实际上输出的仅仅是一个引用, 这里它把a_int当成了一个元素 */
        List a_int_List = Arrays.asList(a_int);
        foreach(a_int_List);
        /* 为此我们需要这样遍历其中元素 */
        foreachForBase(a_int_List);
        System.out.println("1 END 2 START");
        /* 段落二:对象类型的数组使用asList,是我们预期的 */
        Integer[] a_Integer = new Integer[] { 1, 2, 3, 4 };
        List a_Integer_List = Arrays.asList(a_Integer);
        foreach(a_Integer_List);
        System.out.println("2 END 3 START");

        /* 段落三:当更新数组或者asList之后的List,另一个将自动获得更新 */
        a_Integer_List.set(0, 0);
        foreach(a_Integer_List);
        foreach(a_Integer);
        System.out.println("3 END 4 START");

        a_Integer[0] = 5;
        foreach(a_Integer_List);
        foreach(a_Integer);

        /* 段落四:对基本类型数组,通过asList之后的List修改对应的值后,在运行时会报出异常
         * 但是基本类型数组对应的List是会发生变化的,这是毫无疑问的
         */

         a_int_List.set(0, 0);
          foreach(a_int_List);
          foreach(a_int);

        System.out.println("4 END 5 START");
        a_int[0] = 5;
        foreachForBase(a_int_List);
        foreach(a_int);

    }

    /* 打印方法 */
    private static void foreach(List list) {
        for (Object object : list) {
            System.out.print(object + " ");
        }
        System.out.println();

    }

    private static void foreachForBase(List a_int_List) {
        int[] _a_int = (int[]) a_int_List.get(0);
        foreach(_a_int);
    }

    private static void foreach(int[] a_int) {
        for (int i : a_int) {
            System.out.print(i + " ");
        }
        System.out.println();
    }

    private static void foreach(Integer[] _a_Integer) {
        for (int i : _a_Integer) {
            System.out.print(i + " ");
        }
        System.out.println();
    }
}

console:

Exception in thread "main" java.lang.ArrayStoreException: java.lang.Integer
    at java.util.Arrays$ArrayList.set(Arrays.java:3847)
    at com.ysyc.invoicecertify.util.mockservice.AsListTest.main(AsListTest.java:56)
[[email protected]
1 2 3 4
1 END 2 START
1 2 3 4
2 END 3 START
0 2 3 4
0 2 3 4
3 END 4 START
5 2 3 4
5 2 3 4
Disconnected from the target VM, address: ‘127.0.0.1:54490‘, transport: ‘socket‘

Process finished with exit code 1

原文地址:https://www.cnblogs.com/xingzc/p/9144375.html

时间: 2024-10-25 21:45:29

Arrays.asList()vs Collections.singletonList()的相关文章

集合框架(Collection和Collections的区别)

1.Collection: java.util.Collection 是一个集合接口. 它提供了对集合对象进行基本操作的通用接口方法.Collection接口在Java 类库中有很多具体的实现.Collection接口的意义是为各种具体的集合提供了最大化的统一操作方式. 以下接口实现了Collection接口: map,set,list,vector 2.Collections:. java.util.Collections 是一个包装类. 它包含有各种有关集合操作的静态多态方法.此类不能实例化

java中排序函数sort()使用,Arrays.sort()和Collections.sort()

Java中常用的数组或集合排序的方法有两个,一个是java.util.Arrays中的静态方法Arrays.sort(),还有一个是java.util.Collections中的静态方法的Collections.sort()方法,下面分别介绍两种用法. 一.java.util.Arrays中的静态方法Arrays.sort() Arrays中的sort()方法主要是针对各种数据类型(基本数据类型和引用对象类型)的数组元素排序. ....... 关于引用对象类型数组的排序sort()方法要用到接口

为什么使用 Arrays.asList()得到的集合,使用remove( )和 add( )方法会抛出unsupportedoperationexception(不支持操作异常)

这是由于: Arrays.asList() 返回java.util.Arrays$ArrayList, 而不是ArrayList. Arrays$ArrayList和ArrayList都是继承AbstractList,remove,add等 method在AbstractList中是默认throw UnsupportedOperationException而且不作任何操作. ArrayList override这些method来对list进行操作,但是Arrays$ArrayList没有over

黑马程序员-集合框架(Map和Collections)

--Java培训.Android培训.iOS培训..Net培训.期待与您交流!--- 一.概述 Map是一种存储键值对的存储容器,而且保证键的唯一性.提供一种以"键"标识"值"的数据存储方式.接口形式为:Map<K,V>,其中K是此映射所维护的键的类型,V是映射值的类型.其有两个常用子类,HashMap和TreeMap,另有HashTable与HashMap功能类似,是早期版本.三者特点与不同如下: HashMap:JDK1.2版本出现,底层使用哈希表数

初识Java8新特性Lambda(二) 之collections

背景(Background) 如果从一开始就将lambda表达式(闭包)作为Java语言的一部分,那么我们的Collections API肯定会与今天的外观有所不同.随着Java语言获得作为JSR 335一部分的lambda表达式,这具有使我们的Collections接口看起来更加过时的副作用.尽管可能很想从头开始并构建替换的Collection框架(" Collections II"),但是替换Collection框架将是一项主要任务,因为Collections接口遍布JDK库.相反

Java提高篇(三六)-----java集合细节(二):asList的缺陷

在实际开发过程中我们经常使用asList讲数组转换为List,这个方法使用起来非常方便,但是asList方法存在几个缺陷: 一.避免使用基本数据类型数组转换为列表 使用8个基本类型数组转换为列表时会存在一个比较有味的缺陷.先看如下程序: public static void main(String[] args) { int[] ints = {1,2,3,4,5}; List list = Arrays.asList(ints); System.out.println("list'size:&

Java之集合初探(二)Iterator(迭代器),collections,打包/解包(装箱拆箱),泛型(Generic),comparable接口

Iterator(迭代器) 所有实现了Collection接口的容器都有一个iterator方法, 用来返回一个实现了Iterator接口的对象 Iterator对象称作迭代器, 用来方便的实现对容器内的元素的遍历 迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构.迭代器通常被称为"轻量级"对象,因为创建它的代价小. Java中的Iterator功能比较简单,并且只能单向移动: (1) 使用方法iterator()要求容器返回一个I

java.util.Arrays.sort两种方式的排序(及文件读写练习)

import java.io.*; import java.util.*; public class SortTest{ public static void main(String args[]) throws IOException, ClassNotFoundException { FileReader InWord = new FileReader(new File("words.txt")); BufferedReader in = new BufferedReader(In

黑马程序员(Java)---API之集合(Map及其子类、Collections)

------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 5.4 Map及其子类 将键映射到值的对象.一个映射不能包含重复的键:每个键最多只能映射到一个值. Map集合存储元素是成对出现的,Map集合的键是唯一的,值是可重复的.Collection集合存储元素是单独出现的,Collection的子类Set是唯一的,List是可重复的. Map集合的数据结构值针对键有效,跟值无关.Collection集合的数据结构是针对元素有效. Map |--Has