Java中List.of()和Arrays.asList()的区别及原因分析

目录
  • Java中List.of()和Arrays.asList()的区别及原因
    • 1.Arrays.asList()可以插入null
    • 2.用List.of的List自然是不包含null
    • 3.List.of生成的List不能修改
    • 4.关于数组修改对List的影响
    • 原因
  • java listof报错处理

Java中List.of()和Arrays.asList()的区别及原因

动手写一下,让自己更有印象

1.Arrays.asList()可以插入null

而List.of()不可以

import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     List<Integer> ls1 = Arrays.asList(1, 2, null);
     //List<Integer> ls2 = List.of(1,2,null);
     System.out.println(ls1);
     //System.out.println(ls2);
    }
}

/*结果
[1, 2, null]
*/
import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     //List<Integer> ls1 = Arrays.asList(1, 2, null);
     List<Integer> ls2 = List.of(1,2,null);
     //System.out.println(ls1);
     System.out.println(ls2);
    }
}

/*结果
Exception in thread "main" java.lang.NullPointerException.....
*/

2.用List.of的List自然是不包含null

而用Arrays.asList的List包含null。上面结果也可得知。

import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     List<Integer> ls1 = Arrays.asList(1, 2, null);
     //List<Integer> ls2 = List.of(1,2);
     System.out.println(ls1.contains(null));
     //System.out.println(ls2.contains(null));
    }
}

/*结果
true
*/
import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     //List<Integer> ls1 = Arrays.asList(1, 2, null);
     List<Integer> ls2 = List.of(1,2);
     //System.out.println(ls1.contains(null));
     System.out.println(ls2.contains(null));
    }
}

/*结果
Exception in thread "main" java.lang.NullPointerException.....
*/

3.List.of生成的List不能修改

Arrays.asList生成的List能修改。

import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     List<Integer> ls1 = Arrays.asList(1, 2, null);
     //List<Integer> ls2 = List.of(1,2);
     ls1.set(0,5);
     //ls2.set(0,5);
     System.out.println(ls1);
     //System.out.println(ls2);
    }
}

/*结果
[5, 2, null]
*/
import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     //List<Integer> ls1 = Arrays.asList(1, 2, null);
     List<Integer> ls2 = List.of(1,2);
     //ls1.set(0,5);
     ls2.set(0,5);
     //System.out.println(ls1);
     System.out.println(ls2);
    }
}

/*结果
Exception in thread "main" java.lang.UnsupportedOperationExceptio.....
*/

4.关于数组修改对List的影响

数组修改对Arrays.asList生成的List有影响,对List.of 生成的List无影响

import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     Integer[] a = new Integer[]{1,2,3,4};
        // 不能用int[],会导致转型错误,错误: 不兼容的类型: 推论变量 T 具有不兼容的上限
     List<Integer> ls1 = Arrays.asList(a);
     //List<Integer> ls2 = List.of(a);
     System.out.println(ls1);
     //System.out.println(ls2);
     a[0] = 5;
     //ls2.set(0,5);
     System.out.println(ls1);
     //System.out.println(ls2);
    }
}

/*结果
[1, 2, 3, 4]
[5, 2, 3, 4]
*/
import java.util.List;
import java.util.Arrays;
class Solution {
    public static void main(String[] args) {
     Integer[] a = new Integer[]{1,2,3,4};
     //List<Integer> ls1 = Arrays.asList(a);
     List<Integer> ls2 = List.of(a);
     //System.out.println(ls1);
     System.out.println(ls2);
     a[0] = 5;
     //ls2.set(0,5);
     //System.out.println(ls1);
     System.out.println(ls2);
    }
}

/*结果
[1, 2, 3, 4]
[1, 2, 3, 4]
*/

原因

关于List.of为什么不能插入null,和修改了原数组不影响到List.of生成的List。先看看List.of有关的源码

   @SafeVarargs
    @SuppressWarnings("varargs")
    static <E> List<E> of(E... elements) {
        switch (elements.length) { // implicit null check of elements
            case 0:
                return ImmutableCollections.emptyList();
            case 1:
                return new ImmutableCollections.List12<>(elements[0]);
            case 2:
                return new ImmutableCollections.List12<>(elements[0], elements[1]);
            default:
                return new ImmutableCollections.ListN<>(elements);
        }
    }
//---------------------------------------------------------------------------------------
    @Stable
        private final E[] elements;    

    @SafeVarargs
        ListN(E... input) {
            // copy and check manually to avoid TOCTOU
            @SuppressWarnings("unchecked")
            E[] tmp = (E[])new Object[input.length]; // implicit nullcheck of input
            for (int i = 0; i < input.length; i++) {
                tmp[i] = Objects.requireNonNull(input[i]);
            }
            elements = tmp;
        }

//---------------------------------------------------------------------------------------

    public static <T> T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }

可以看到Objects.requireNonNull()。所以不能插入空值。

E[] tmp = (E[])new Object[input.length];表示新建了个新的数组对象,所以修改了原数组,不影响生成的LIst底层的数组。

返回的数组是个final类型的,所以不能修改

再看看Arrays.asList源码

    @SafeVarargs
    @SuppressWarnings("varargs")
    public static <T> List<T> asList(T... a) {
        return new ArrayList<>(a);
    }

//----------------------------------------------------------------------------------------

    ArrayList(E[] array) {
            a = Objects.requireNonNull(array);
        }

//----------------------------------------------------------------------------------------

    public static <T> T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }

由源码可知,底层的数组就是传入的数组,所以对原数组的修改会影响到用Arrays.asList方法生成的List。而且Objects.requireNonNull(array)检查的是整个数组是不是null,而非对每个元素进行检查是否为null。所以用Arrays.asList方法可以插入空值。

也没有规定是final的,所以支持修改。

java listof报错处理

List.of()生成不可变数组(字符串也行)

是在jdk1.8以后才出现的,在jdk1.9版本及以后才能运行。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

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

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

  • 浅谈Arrays.asList() 和ArrayList类型区别

    <pre name="code" class="html"><pre name="code" class="html">Arrays.asList() 将一个数组转化为一个List对象,这个方法会返回一个ArrayList类型的对象, 这个ArrayList类并非java.util.ArrayList类,而是Arrays类的静态内部类!用这个对象对列表进行添加删除更新操作,就会报UnsupportedO

  • java 源码分析Arrays.asList方法详解

    最近,抽空把java Arrays 工具类的asList 方法做了源码分析,在网上整理了相关资料,记录下来,希望也能帮助读者! Arrays工具类提供了一个方法asList, 使用该方法可以将一个变长参数或者数组转换成List . 其源代码如下: /** * Returns a fixed-size list backed by the specified array. (Changes to * the returned list "write through" to the arr

  • Java中Arrays.asList()方法详解及实例

    Arrays.asList() 是将数组作为列表. 问题来源于: public class Test { public static void main(String[] args) { int[] a = {1, 2, 3, 4}; List list = Arrays.asList(a); System.out.println(list.size()); //1 } } 期望的输出是 list 里面也有4个元素,也就是 size 为4,然而结果是1. 原因如下: 在 Arrays.asLis

  • Java Arrays.asList使用方法解析

    Arrays.asList()方法的作用是将数组或一些元素转为集合,而你得到的集合并不是我们通常使用的List集合,而是Arrays里面的一个内部类.阿里的开发手册上java开发规范说到使用工具类Arrays.asList()方法把数组转换成集合时,不能使用其修改集合相关的方法,它的add/remove/clear方法会抛出java.lang.UnsupportedOperationException的异常. 一.Arrays.asList的方法说明 public static void mai

  • Java中List.of()和Arrays.asList()的区别及原因分析

    目录 Java中List.of()和Arrays.asList()的区别及原因 1.Arrays.asList()可以插入null 2.用List.of的List自然是不包含null 3.List.of生成的List不能修改 4.关于数组修改对List的影响 原因 java listof报错处理 Java中List.of()和Arrays.asList()的区别及原因 动手写一下,让自己更有印象 1.Arrays.asList()可以插入null 而List.of()不可以 import jav

  • Java踩坑记录之Arrays.AsList

    前言 java.util.Arrays的asList方法可以方便的将数组转化为集合,我们平时开发在初始化ArrayList时使用的比较多,可以简化代码,但这个静态方法asList()有几个坑需要注意: 一. 如果对集合使用增加或删除元素的操作将会报错 如下代码: List list = Arrays.asList("a","b","c"); list.add("d"); 输出结果: Exception in thread &q

  • Java中操作数组的Arrays类

    引包:位于java.util 包下. Arrays类包含了各种操作数组的静态方法: 数组排序:sort(升序排序)重载了各种数组升序排序方法,举例几种: sort(char[] a)sort(double[] a)sort(int[] a) 示例: public class ArraysTest { public static void main(String[] args) { int[] arrInt = {9,8,7,6,5,4,3,2,1}; Arrays.sort(arrInt); f

  • java中list.forEach()和list.stream().forEach()区别

    目录 概述 区别 首先,它们的功能都是遍历数组每个元素并执行入参的accept()方法,但是它们的实现方式却不一样,在一些特定的情况下,执行会出现不同的结果. 在大多数情况下,两者都会产生相同的结果,但是,我们会看到一些微妙的差异. 概述 首先,创建一个迭代列表: List<String> list = Arrays.asList("A","B","C","D"); 最直接的方法是使用增强的for循环: for(S

  • Java中stream.map和stream.forEach的区别

    目录 什么是 stream 流 stream.map 和 stream.forEach 的区别 网上很多关于讲解这俩个区别的文章,但大多数要么不明不白,要么太复杂难理解.所以自己通俗的讲一下,毕竟不会太深奥,只是个人理解 (评论区指出了错误改了一下). 什么是 stream 流 我们在使用集合或数组对元素进行操作时往往会遇到这种情况:通过对不同类型的存储元素,按照特定条件进行查找.排序.等操作时往往会写一大段代码,而且更要命的是,不同类型的数据,操作的方法也不一样,比如一个存储 Student

  • Java中List Set和Map之间的区别_动力节点Java学院整理

    Java集合的主要分为三种类型: • Set(集) • List(列表) • Map(映射) 要深入理解集合首先要了解下我们熟悉的数组: 数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型),而JAVA集合可以存储和操作数目不固定的一组数据. 所有的JAVA集合都位于 java.util包中! JAVA集合只能存放引用类型的的数据,不能存放基本数据类型. 世间上本来没有集合,(只有数组参考C语言)但有人想要,所以有了集合 有人想有可以自动扩展的数组,所以有了List 有的

  • Java中List  Set和Map之间的区别_动力节点Java学院整理

    Java集合的主要分为三种类型: • Set(集) • List(列表) • Map(映射) 要深入理解集合首先要了解下我们熟悉的数组: 数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型),而JAVA集合可以存储和操作数目不固定的一组数据. 所有的JAVA集合都位于 java.util包中! JAVA集合只能存放引用类型的的数据,不能存放基本数据类型. 世间上本来没有集合,(只有数组参考C语言)但有人想要,所以有了集合 有人想有可以自动扩展的数组,所以有了List 有的

  • 详谈Java中的Object、T(泛型)、?区别

    因为最近重新看了泛型,又看了些反射,导致我对Object.T(以下代指泛型).?产生了疑惑. 我们先来试着理解一下Object类,学习Java的应该都知道Object是所有类的父类,注意:那么这就意味着它的范围非常广!首先记住这点,如果你的参数类型时Object,那么的参数类型将非常广! <Thinking in Java>中说很多原因促成了泛型的出现,最引人注目的一个原因就是为了创造容器类.这个要怎么来理解呢?我的理解是,可以抛开这个为了创造容器类这个,而是回到泛型的目的是限定某种类型上来.

  • java 中newInstance()方法和new关键字的区别

    java 中newInstance()方法和new关键字的区别 * 它们的区别在于创建对象的方式不一样,前者是使用类加载机制,后者是创建一个新类. * 那么为什么会有两种创建对象方式?这主要考虑到软件的可伸缩.可扩展和可重用等软件设计思想. * 我们使用关键字new创建一个类的时候,这个类可以没有被加载.但是使用newInstance()方法的时候, * 就必须保证:1.这个类已经加载:2.这个类已经连接了. * newInstance()实际上是把new这个方式分解为两步,即首先调用Class

  • 浅谈Java中的四种引用方式的区别

    强引用.软引用.弱引用.虚引用的概念 强引用(StrongReference) 强引用就是指在程序代码之中普遍存在的,比如下面这段代码中的object和str都是强引用: Object object = new Object(); String str = "hello"; 只要某个对象有强引用与之关联,JVM必定不会回收这个对象,即使在内存不足的情况下,JVM宁愿抛出OutOfMemory错误也不会回收这种对象. 比如下面这段代码: public class Main { publi

随机推荐