好用的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. MySQL锁会不会,你就差看一看

    数据库锁知识 不少人在开发的时候,应该很少会注意到这些锁的问题,也很少会给程序加锁(除了库存这些对数量准确性要求极高的情况下),即使我们不会这些锁知识,我们的程序在一般情况下还是可以跑得好好的.因为这 ...

  2. php+redis实现注册、删除、编辑、分页、登录、关注等功能

    本文实例讲述了php+redis实现注册.删除.编辑.分页.登录.关注等功能.分享给大家供大家参考,具体如下: 主要界面 ​ 连接redis redis.php <?php //实例化 $red ...

  3. fastjson自定义序列化竟然有这么多姿势?

    本文介绍下fastjson自定义序列化的各种操作. 一.什么是fastjson? fastjson是阿里巴巴的开源JSON解析库,它可以解析JSON格式的字符串,支持将Java Bean序列化为JSO ...

  4. Java之Retry重试机制详解

    应用中需要实现一个功能: 需要将数据上传到远程存储服务,同时在返回处理成功情况下做其他操作.这个功能不复杂,分为两个步骤:第一步调用远程的Rest服务上传数据后对返回的结果进行处理:第二步拿到第一步结 ...

  5. kubeadm配置高可用etcd集群

    操作系统为ubuntu18 kubernetes版本为v1.15.1 k8s默认在控制平面节点上的kubelet管理的静态pod中运行单个成员的etcd集群,但这不是高可用的方案. etcd高可用集群 ...

  6. while(cin)?

    #include<iostream> #include<utility> using namespace std; int main() { int i; do { cout& ...

  7. Oracle temp table

    Tow kinds of temp table data keep method. One is delete when commit Anothe one is preseve when commi ...

  8. 网络ASI

    ASIHTTPRequest  基于底层CFNetwork框架,运行效率很高 可惜作者 停止更新,有一些潜在的BUG无人去解决 老项目 ASI + SBJson 只需要用到外面的源文件 ASI还依赖于 ...

  9. 查看/设置JVM使用的垃圾收集器

    一.设置垃圾收集器的参数 -XX:+UseSerialGC,虚拟机在Client模式下的默认值,Serial+Serial Old -XX:+UseParNewGC,ParNew+Serial Old ...

  10. PHP计算两组经纬度坐标之间的距离

    定义π define('PI',3.1415926535898); define('EARTH_RADIUS',6378.137); 计算两组经纬度坐标 之间的距离 /** * 计算两组经纬度坐标 之 ...