好用的asList

在开发或写测试用例的过程中,经常会用到Arrays.asList()这个方法,可以快速方便地将数组转化成一个List。例如:

List<String> list = Arrays.asList("Book", "Pen", "Desk", "Cup");

当我们静态引用Arrays.asList()后:

import static java.util.Arrays.asList;

可以直接这样写:

List<String> list = asList("Book", "Pen", "Desk", "Cup");

隐藏的坑

基本类型不可泛型化

执行下面测试用例:

@Test
public void size() {
int[] nums = {1, 2, 3, 4, 5, 6};
List list = asList(nums);
assertEquals(nums.length, list.size());
}

结果为failed

java.lang.AssertionError:
Expected :6
Actual :1

为什么明明是6个元素的数组,转化为List后便只有一个元素呢?

源码是不会说谎的,让我们来看看代码:

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

通过源码可以得知asList()方法的入参为泛型,对int这种基本类型,是无法泛型化的,所以函数把整个数组当成了一个整体(数组为引用类型,可以泛型化)。最终返回的结果是List<int[]>,而不是List<Integer>

如果我们需要List<Integer>,可以用下面的两种方法来处理:

@Test
public void listForInt() {
//方法1:初始化为Integer的数组,初始化时自动装箱
Integer[] nums = {1, 2, 3, 4, 5, 6};
List<Integer> list = asList(nums);
assertEquals(nums.length, list.size());
//方法2:不传入整体,处理参数时自动装箱
list = asList(1, 2, 3, 4, 5, 6);
assertEquals(6, list.size());
}

以上两种方法,返回的结果都是List<Integer>了。

不可修改

高高兴兴转化成了List,正准备大干一场,进行List的常规操作了,却发现操作不得:

@Test
public void listAdd() {
List<String> list = asList("Book", "Pen", "Desk", "Cup");
list.add("Box");
assertEquals(5, list.size());
}

结果报错如下:

java.lang.UnsupportedOperationException
at java.util.AbstractList.add(AbstractList.java:148)
at java.util.AbstractList.add(AbstractList.java:108)
at com.larry.basic.AsListTest.listAdd(AsListTest.java:42)

只好再次翻看源码得知,虽然asList()方法返回的结果是ArrayList,但与我们平常用的ArrayList却是不一样的:

我们平常用的最多的是java.util.ArrayList,底层为可变数组的List。而java.util.Arrays.ArrayList是Arrays的一个静态内部类,底层为final的数组的List。他们并不是同一个类。

java.util.Arrays.ArrayList没有重写add/remove/clear等方法,因此会调用父类AbstractList的方法,而父类的方法如下:

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();
}

所以,这些方法实际上是不可调用的,会抛异常UnsupportedOperationException

修改操作set的副作用

asList()的结果真的是不可修改的吗?其实也不是。虽然Arrays.ArrayList没有重写add/remove/clear方法,但重写了set()方法:

@Override
public E set(int index, E element) {
E oldValue = a[index];
a[index] = element;
return oldValue;
}

我们可以对其中的元素进行替换。这其实很好理解的,底层为final的数组,大小不可变,但数组的元素可变。因为有这个功能,可能会引发下面的问题:

@Test
public void listSet() {
String[] arr = {"Book", "Pen", "Desk", "Cup"};
List<String> list = asList(arr);
list.set(0, "New Book");
assertEquals("New Book", list.get(0));
assertEquals("Book", arr[0]);
}

代码最后一句报错了,当改变了List的第一个元素,数组的第一个元素也被改了,因为它们都指向了同一个数组地址。稍不注意,就会生产与期待不同的结果。

如果要新建一个List,可以采用下面的方法:

List<String> list = new ArrayList<String>(asList(arr));

因为new ArrayList()时会用方法Arrays.copyOf()复制一份新的数组出来。

总结

简单常用的东西,也要小心谨慎。


欢迎关注公众号<南瓜慢说>,将为你持续更新...

【Java必修课】好用的Arrays.asList也有这三个坑的更多相关文章

  1. Java数组转集合之Arrays.asList()用法

    Arrays.asList()用法 使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合. 而一开始asList的设计时用于打印数组而设计 ...

  2. java Arrays.asList

    List<String> list = Arrays.asList("A B C D E F G H I J K L ".split(" ")); ...

  3. Arrays.asList()后调用add,remove这些method时出现java.lang.UnsupportedOperationException异常

    String[] queryNames = request.getParameterValues("queryName"); List<String> queryNam ...

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

    在平时的开发过程中,我们知道能够将一个Array的对象转化为List.这种操作,我们仅仅要採用Arrays.asList这种方法即可了.笔者前段时间一直用这种方法,有一天,我发现通过Arrays.as ...

  5. java之Arrays.asList

    使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合. 而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比 ...

  6. Arrays.asList()注意

    api: public static <T> List<T> asList(T... a) 返回一个受指定数组支持的固定大小的列表.(对返回列表的更改会“直接写”到数组.)此方 ...

  7. 【转】java.util.Arrays.asList 的用法

    DK 1.4对java.util.Arrays.asList的定义,函数参数是Object[].所以,在1.4中asList()并不支持基本类型的数组作参数. JDK 1.5中,java.util.A ...

  8. java Arrays.asList()和Collections.addAll()

    java中的方法Arrays.asList(arg1,arg2,arg3...),经常用在将多个元素或数组转化为List中的元素,但是在使用的时候,应该注意: arg1决定返回list的元素类型(即第 ...

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

    总结 利用Arrays.asList方法返回的List是不允许add和remove的,这种list的长度不可变,因为底层依然是写数组. Arrays.asList的返回值是调用是传入T类型的List, ...

随机推荐

  1. Laravel 菜鸟的晋级之路

    第一阶段:简单的增删改查 这是最开始接触Laravel的一个阶段.如果有PHP经验,那么应该能很快找到MVC的路径,然后驾轻就熟的开始写起来.虽然还显得有些笨拙,不过很快就能做出一些内容了.如果没有P ...

  2. Matlab查看本机IP地址---xdd

    复制粘贴于http://www.matlabsky.com/thread-28597-1-1.html [s, r]=system('ipconfig') % r=regexp(r,'IP Addre ...

  3. 分布式远程调用SpringCloud-Feign的两种具体操作方式(精华)

    一 前言 几大RPC框架介绍 1.支持多语言的RPC框架,google的gRPC,Apache(facebook)的Thrift 2.只支持特定语言的RPC框架,例如新浪的Motan 3.支持服务治理 ...

  4. CentOS 7 Nginx部署.NET Core Web应用

    部署.NET Core运行时 必要前提 在安装.NET Core前,需要注册Microsoft签名秘钥并添加Microsoft产品提要,每台机器只需要注册一次,执行如下命令: sudo rpm -Uv ...

  5. react新版本配置代理

    新学习react 开始配置react跨域的时候 在网上查看到是在packjson.json里面添加如下代码: "proxy": { "/api": { &quo ...

  6. Docker虚拟化之<基础命令>

    1.在docker hub中搜索镜像 docker search nginx 2.从docker镜像服务器拉取指定镜像或者库镜像 docker pull docker.io/nginx 3.列出系统当 ...

  7. Nginx+MySQL+PHP+Redis多机部署(测试发布discuz论坛)

    链接:LNMP+Redis单机部署 1.实战多机部署环境 nginx服务器: 192.168.1.3 php服务器:    192.168.1.4 mysql服务器: 192.168.1.10 red ...

  8. wdCP v3.3.8apache阿里云ssl证书实现智慧软文http转换https的详细操作教程

    先展示一下效果:智慧软文发布系统(https://www.zhihuiruanwen.com) 之前用的是传统的http,发现360浏览器,火狐浏览器,谷歌浏览器均提示不安全的链接,最主要的是第一次打 ...

  9. 第八次作业-非确定的自动机NFA确定化为DFA

    NFA 确定化为 DFA 子集法: f(q,a)={q1,q2,…,qn},状态集的子集 将{q1,q2,…,qn}看做一个状态A,去记录NFA读入输入符号之后可能达到的所有状态的集合. 步骤: 1. ...

  10. 每个开发人员都应该知道的11个Linux命令

    本文主要挑选出读者有必要首先学习的 11 个 Linux 命令,如果不熟悉的读者可以在虚拟机或云服务器上实操下,对于开发人员来说,能熟练掌握 Linux 做一些基本的操作是必要的! 事不宜迟,这里有 ...