【Java必修课】好用的Arrays.asList也有这三个坑
好用的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也有这三个坑的更多相关文章
- Java数组转集合之Arrays.asList()用法
Arrays.asList()用法 使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合. 而一开始asList的设计时用于打印数组而设计 ...
- java Arrays.asList
List<String> list = Arrays.asList("A B C D E F G H I J K L ".split(" ")); ...
- Arrays.asList()后调用add,remove这些method时出现java.lang.UnsupportedOperationException异常
String[] queryNames = request.getParameterValues("queryName"); List<String> queryNam ...
- 为什么Java里的Arrays.asList不能用add和remove方法?
在平时的开发过程中,我们知道能够将一个Array的对象转化为List.这种操作,我们仅仅要採用Arrays.asList这种方法即可了.笔者前段时间一直用这种方法,有一天,我发现通过Arrays.as ...
- java之Arrays.asList
使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合. 而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比 ...
- Arrays.asList()注意
api: public static <T> List<T> asList(T... a) 返回一个受指定数组支持的固定大小的列表.(对返回列表的更改会“直接写”到数组.)此方 ...
- 【转】java.util.Arrays.asList 的用法
DK 1.4对java.util.Arrays.asList的定义,函数参数是Object[].所以,在1.4中asList()并不支持基本类型的数组作参数. JDK 1.5中,java.util.A ...
- java Arrays.asList()和Collections.addAll()
java中的方法Arrays.asList(arg1,arg2,arg3...),经常用在将多个元素或数组转化为List中的元素,但是在使用的时候,应该注意: arg1决定返回list的元素类型(即第 ...
- 【Java基础】一个有意思的泛型方法Arrays.asList(T... a)
总结 利用Arrays.asList方法返回的List是不允许add和remove的,这种list的长度不可变,因为底层依然是写数组. Arrays.asList的返回值是调用是传入T类型的List, ...
随机推荐
- pwnable.kr第二天
3.bof 这题就是简单的数组越界覆盖,直接用gdb 调试出偏移就ok from pwn import * context.log_level='debug' payload='A'*52+p32(0 ...
- Gemini.Workflow 双子工作流高级教程:数据库设计及各表作用说明
整体数据库设计,可见这一篇:Gemini.Workflow 双子工作流高级教程:数据库-设计文档 这里对各数据表进行介绍: 工作流里的设计表并不多,核心只有以下8个: 下面按照流程的顺序来介绍一下表的 ...
- 宋宝华:关于ARM Linux原子操作的实现
本文系转载,著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 作者: 宋宝华 来源: 微信公众号linux阅码场(id: linuxdev) 竞态无所不在 首先我们要理解竞态(ra ...
- 3sql
-------------------- 三范式-- 第一范式(1NF):强调的是列的原子性,即列不能够再分成其他几列. -- 第二范式(2NF):首先是 1NF,另外包含两部分内容,一是表必须有一个 ...
- 小白学 Python 爬虫(9):爬虫基础
人生苦短,我用 Python 前文传送门: 小白学 Python 爬虫(1):开篇 小白学 Python 爬虫(2):前置准备(一)基本类库的安装 小白学 Python 爬虫(3):前置准备(二)Li ...
- 使用Cap解决.Netcore分布式事务
一.什么是Cap CAP 是一个基于 .NET Standard 的 C# 库,它是一种处理分布式事务的解决方案,同样具有 EventBus 的功能,它具有轻量级.易使用.高性能等特点. 在我们 ...
- Logistic回归算法梯度公式的推导
最近学习Logistic回归算法,在网上看了许多博文,笔者觉得这篇文章http://blog.kamidox.com/logistic-regression.html写得最好.但其中有个关键问题没有讲 ...
- Thinkphp5——实现分页(模型和Db分页,多种方法)
现在很多网站的数据量的很多,如果全部在一页里显示效果不好,数据量太大,那怎么办?这时我们就需要分页,而分页的好处就是分段显示数据,这样页面就不用加载很多数据,需要时才加载,下面我教大家实现ThinkP ...
- 【Python成长之路】python 基础篇 -- 装饰器【华为云分享】
[写在前面] 有时候看到大神们的代码,偶尔会用到@来装饰函数.当时查了资料,大致了解装饰器一般用于在不改变原函数的基础上 ,对原函数功能进行修改/增强.使用场景是:日志级别设置.权限校验.性能测试等. ...
- python让你再也不为文章配图与素材发愁,让高清图片占满你的硬盘! #华为云·寻找黑马程序员#
欢迎添加华为云小助手微信(微信号:HWCloud002 或 HWCloud003),输入关键字"加群",加入华为云线上技术讨论群:输入关键字"最新活动",获取华 ...