1. 概述

在使用List集合时有些地方需要注意一下的, 不然会出现一些莫名其妙的错误.

2. Arrays.asList();

2-1. 产生不可操作的集合

来看一个例子.

  1. Integer[] array = {1, 2, 3, 4, 5};
  2. List<Integer> list = Arrays.asList(array);
  3. list.remove(1);

会直接抛出java.lang.UnsupportedOperationException异常, 为什么呢?

答案是: 返回的List非java.util.ArrayList. 而是在Arrays类内部有一个静态内部类叫ArrayList.

静态内部类ArrayList并没有提供所有的方法实现, 有些方法在抽象类中仅仅是一个抛异常的实现, 上述的remove就是这种情况. asList返回的列表只不过是披着list的外衣, 它并没有list的基本特性(变长), 该list是一个长度不可变的列表, 传入参数的数组有多长, 其返回的列表就只能是多长. 所以不要试图改变asList的返回列表.

Arrays.asList的代码如下:

  1. public static <T> List<T> asList(T... a) {
  2. return new ArrayList<>(a);
  3. }
  4. private static class ArrayList<E> extends AbstractList<E>
  5. implements RandomAccess, java.io.Serializable
  6. {
  7. private static final long serialVersionUID = -2764017481108945198L;
  8. private final E[] a;
  9. ArrayList(E[] array) {
  10. a = Objects.requireNonNull(array);
  11. }
  12. // 省略部分源码
  13. }

该类也是继承了AbstractkList抽象类, 只是该类并没有提供所有方法的实现.

2-2. 需要传入对象数组而不是基本类型数组

看个例子:

  1. int[] array1 = {1, 2, 3, 4, 5};
  2. List<int[]> list1 = Arrays.asList(array1);
  3. System.out.println(list1.size()); // 1
  4. Integer[] array2 = {1, 2, 3, 4, 5};
  5. List<Integer> list2 = Arrays.asList(array2);
  6. System.out.println(list2.size()); // 5

看到Array.asList的返回值的泛型就明白, list1中存储的实际是一个数组. 为什么这样呢? 数值类型不能被泛型化, 比如List<int>是不对的, 必须是List<Integer>才对.

3. arrayList.subList();

3-1. subList返回的并不是ArrayList

方法返回的是一个内部类SubList, 并不是ArrayList, SubList也是继承AbstractList, 内部引用了原列表的List.

3-2. subList返回的仅仅是一个视图

subList返回的仅仅是一个视图, 对subList返回值的所有操作都会作用在原列表中.

看一个例子

  1. List<String> list = new ArrayList<>();
  2. list.add("A");
  3. list.add("B");
  4. list.add("C");
  5. list.add("D");
  6. list.add("E");
  7. List<String> subList = list.subList(2, 4);
  8. subList.add("F");
  9. System.out.println(list); // 输出 [A, B, C, D, F, E]

通过输出结果可知, F被添加到原列表中了.

3-3. 生成子列表时, 不要试图去操作原列表

因为子列表是根据原列表动态生成的, 所以如果原列表变了, 那么操作子列表时就会发生ConcurrentModificationException异常. 这个异常的原因相信你也知道, 就是modCount改变了导致的.

3-4. 推荐使用subList处理局部链表

比如, 有一个需求要删除列表100-200位置处的数据, 通常我们这样写:

  1. int start = 100;
  2. int end = 200;
  3. for (int i = start; i <= end; i++) {
  4. list.remove(start);
  5. }

通过subList, 我们有一个更简洁的处理方式:

  1. list.subList(100, 200).clear();

简洁.

不定期更新

可能还会遇到集合相关的坑.

随机推荐

  1. CentOS上升级gcc编译器使支持C++11

    首先向博主致敬,好的东西拿来共享了,用一下不错. https://blog.csdn.net/clirus/article/details/62424517 0. 目标  最近在学习c++11,我本机 ...

  2. C# 网络通信功能 同步数据交互开发

    前言 本文将使用一个Nuget公开的组件技术来实现一对多的数据通信功能,提供了一些简单的API,来方便的向服务器进行数据请求. 在visual studio 中的Nuget管理器中可以下载安装,也可以 ...

  3. python 参数传递 传值还是传引用

    个人推测结论: 可变对象传引用,不可变对象传值 python里的变量不同于c中地址储值模型 a=100 b=100 print(id(a),id(b),a==b,a is b) #8790877986 ...

  4. 深入理解synchronized方法同步的是方法还是对象?

    一.运用synchronized关键字 首先我们来看看一个多线程中线程不安全的列子 代码如下: 共享数据类: public class NotSynchronizated extends Thread ...

  5. Spark访问Hive表

    知识点1:Spark访问HIVE上面的数据 配置注意点:. 1.拷贝mysql-connector-java-5.1.38-bin.jar等相关的jar包到你${spark_home}/lib中(sp ...

  6. Python全栈之路---数据类型—字符串

    字符串:有序的字符的集合,用于存储和表示基本的文本信息,一对单.双.或三引号中间包含的内容称之为字符串 1.特性:有序,不可变(开辟新地址存储字符串,python解释器会定期清空不用了的已存储的) & ...

  7. Elasticsearch5.5.1学习笔记

    在linux下增加ik分词 一.下载分词器安装包 wget https://github.com/medcl/elasticsearch-analysis-ik/releases/download/v ...

  8. 2018.4.26 lvm

    lvm(Logical Volume Manager)逻辑卷管理,是Linux环境下对磁盘分区进行管理的一种机制. 基本概念: 1. 物理卷-----PV(Physical Volume)物理卷在逻辑 ...

  9. Ubuntu16.04交叉工具链安装

    前言: 开发环境是64位的ubuntu16.04,交叉工具链是通过sudo apt-get install ....安装的,移植uboot2014.10,但是很奇怪,按照网上的介绍在start.s里面 ...

  10. jquery中使元素显示和隐藏方法之间的区别

    在实际的项目开发中,要使一个元素隐藏的方法有很多,比如css的多种属性和jquery的多种方法,虽然他们的作用都是使元素不可见,但是各个方法实现的原理是不一样的.下面主要介绍jquery各个元素隐藏方 ...