Java中好用的Arrays.asList有哪些坑

发布时间:2021-11-15 15:38:10 作者:iii
来源:亿速云 阅读:151

这篇文章主要介绍“Java中好用的Arrays.asList有哪些坑”,在日常操作中,相信很多人在Java中好用的Arrays.asList有哪些坑问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”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有哪些坑”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注亿速云网站,小编会继续努力为大家带来更多实用的文章!

推荐阅读:
  1. Arrays.asList()数组转集合的坑
  2. Java CPP的坑有哪些

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

java

上一篇:http2中的首部压缩的实现原理是什么

下一篇:Centos7.6如何部署ELK日志分析系统

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》