原来Java中有两个ArrayList

首先给出一段代码:

public class AslistMethod {
 public static void main(String[] args) {
  String sentence = "i love you";
  List<String> words = Arrays.asList(sentence.split(" "));
  for (int i = 0; i < words.size(); ) {
   if (words.get(i).equals("love")) {
    words.remove(i);
    continue;
   }
   i++;
  }
  System.out.println(words);
 }

然后,运行这段代码,最后,报错了:

Exception in thread "main" java.lang.UnsupportedOperationException
 at java.util.AbstractList.remove(AbstractList.java:161)
 at list.AslistMethod.main(AslistMethod.java:12)

错误链的栈底list.AslistMethod.main(AslistMethod.java:12)说的是这一行代码:

words.remove(i);

这个错误让我很迷茫,因为我写这段代码前大概看了一下,Arrays.asList方法返回的是一个ArrayList,咋就不能remove了呢?接下来,由于一个小小的失误,把Arrays.asList的结果赋给了java.util.ArrayList,居然赋值不成功,我立马有种瞎猫碰上死耗子的感觉。然后,就顺着Arrays.asList方法一直点下去,发现此ArrayList非彼ArrayList。Arrays.asList返回的是java.util.Arrays.ArrayList,它只是Arrays的一个内部类。它的代码不长,这里直接贴出来:

 private static class ArrayList<E> extends AbstractList<E>
  implements RandomAccess, java.io.Serializable
 {
  private static final long serialVersionUID = -2764017481108945198L;
  private final E[] a;

  ArrayList(E[] array) {
   a = Objects.requireNonNull(array);
  }

  @Override
  public int size() {
   return a.length;
  }

  @Override
  public Object[] toArray() {
   return a.clone();
  }

  @Override
  @SuppressWarnings("unchecked")
  public <T> T[] toArray(T[] a) {
   int size = size();
   if (a.length < size)
    return Arrays.copyOf(this.a, size,
          (Class<? extends T[]>) a.getClass());
   System.arraycopy(this.a, 0, a, 0, size);
   if (a.length > size)
    a[size] = null;
   return a;
  }

  @Override
  public E get(int index) {
   return a[index];
  }

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

  @Override
  public int indexOf(Object o) {
   if (o==null) {
    for (int i=0; i<a.length; i++)
     if (a[i]==null)
      return i;
   } else {
    for (int i=0; i<a.length; i++)
     if (o.equals(a[i]))
      return i;
   }
   return -1;
  }

  @Override
  public boolean contains(Object o) {
   return indexOf(o) != -1;
  }

  @Override
  public Spliterator<E> spliterator() {
   return Spliterators.spliterator(a, Spliterator.ORDERED);
  }
 }

从源码看到,它继承自AbstractList,并且没有remove方法,我找到AbstractList的remove方法:

 public E remove(int index) {
  throw new UnsupportedOperationException();
 }

这就是错误信息的来源。最终知道,Arrays.asList返回的ArrayList并不支持remove,实际上,java.util.Arrays.ArrayList不支持任何形式的元素删除。
总结一下,这次问题的解决确实有点儿瞎猫碰到死耗子的味道,但是,只要我认真看一下报的错误:

at java.util.AbstractList.remove(AbstractList.java:161)

调用java.util.AbstractList的remove方法和我预期的不一样,它就能清晰地告诉我方向。

希望本文所述对大家学习java程序设计有所帮助。

(0)

相关推荐

  • Java ArrayList.toArray(T[]) 方法的参数类型是 T 而不是 E的原因分析

    前两天给同事做 code review,感觉自己对 Java 的 Generics 掌握得不够好,便拿出 <Effective Java>1 这本书再看看相关的章节.在 Item 24:Eliminate unchecked warnings 这一节中,作者拿 ArrayList 类中的 public <T> T[] toArray(T[] a) 方法作为例子来说明如何对变量使用 @SuppressWarnings annotation. ArrayList 是一个 generic

  • java ArrayList集合中的某个对象属性进行排序的实现代码

    开发中有时候需要自己封装分页排序时,List如何对某一属性排序呢,分享一个小实例,大家共勉,希望能对大家有用,请多多指教. 1.Student的Bean如下: public class Student { private int age; private String name; private String weight; public String getWeight() { return weight; } public void setWeight(String weight) { th

  • Java中ArrayList类的用法与源码完全解析

    System.Collections.ArrayList类是一个特殊的数组.通过添加和删除元素,就可以动态改变数组的长度. 一.优点 1. 支持自动改变大小的功能 2. 可以灵活的插入元素 3. 可以灵活的删除元素 二.局限性 跟一般的数组比起来,速度上差些 三.添加元素 1.publicvirtualintAdd(objectvalue); 将对象添加到ArrayList的结尾处 ArrayList aList = new ArrayList(); aList.Add("a"); a

  • 分析Java中ArrayList与LinkedList列表结构的源码

    一.ArrayList源码分析(JDK7) ArrayList内部维护了一个动态的Object数组,ArrayList的动态增删就是对这个对组的动态的增加和删除. 1.ArrayList构造以及初始化 ArrayList实例变量 //ArrayList默认容量 private static final int DEFAULT_CAPACITY = 10; //默认空的Object数组, 用于定义空的ArrayList private static final Object[] EMPTY_ELE

  • 由ArrayList来深入理解Java中的fail-fast机制

    1. fail-fast简介 "快速失败"也就是fail-fast,它是Java集合的一种错误检测机制.某个线程在对collection进行迭代时,不允许其他线程对该collection进行结构上的修改. 例如:假设存在两个线程(线程1.线程2),线程1通过Iterator在遍历集合A中的元素,在某个时候线程2修改了集合A的结构(是结构上面的修改,而不是简单的修改集合元素的内容),那么这个时候程序就会抛出 ConcurrentModificationException 异常,从而产生f

  • Java中ArrayList和LinkedList的遍历与性能分析

    前言 通过本文你可以了解List的五种遍历方式及各自性能和foreach及Iterator的实现,加深对ArrayList和LinkedList实现的了解.下面来一起看看吧. 一.List的五种遍历方式 1.for each循环 List<Integer> list = new ArrayList<Integer>(); for (Integer j : list) { // use j } 2.显示调用集合迭代器 List<Integer> list = new Ar

  • java正则表达式实现提取需要的字符并放入数组【ArrayList数组去重复功能】

    本文实例讲述了java正则表达式实现提取需要的字符并放入数组.分享给大家供大家参考,具体如下: 这里演示Java正则表达式提取需要的字符并放入数组,即ArrayList数组去重复功能. 具体代码如下: package com.test.tool; import java.util.ArrayList; import java.util.HashSet; import java.util.regex.*; public class MatchTest { public static void ma

  • java 对ArrayList进行分页实例代码

    java 对ArrayList进行分页 概述 系统与系统之间的交互,通常是使用接口的形式.假设B系统提供了一个批量的查询接口,限制每次只能查询50条数据,而我们实际需要查询500条数据,这个时候可以对这500条数据做分批操作,分10次调用B系统的批量接口. 如果B系统的查询接口是使用List作为入参,那么要实现分批调用的话,可以利用ArrayList的subList方法来处理. 代码 sublist方法的定义: List<E> subList(int fromIndex, int toInde

  • java Vector和ArrayList的分析及比较

     java Vector和ArrayList 比较 今天研究了一下Vector和ArrayList的源码,又加深了对这两个类的理解. List接口下一共实现了三个类:ArrayList,Vector,LinkedList.LinkedList就不多说了,它一般主要用在保持数据的插入顺序的时候. ArrayList和Vector都是用数组实现的,主要有这么三个区别: 1.Vector是多线程安全的,而ArrayList不是,这个可以从源码中看出,Vector类中的方法很多有synchronized

  • Java ArrayList 实现实例讲解

     ArrayList概述:  ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存. ArrayList不是线程安全的,只能用在单线程环境下,多线程环境下可以考虑用Collections.synchronizedList(List l)函数返回一个线程安全的ArrayList类,也可以使用concurrent并发包下的CopyOnWriteArrayList类. ArrayList实现了Serializable接口,因此它支持序列化,能

随机推荐