ArrayList.remove()的两种重载:按索引删,还是按对象删?

在Ja va集合框架里,ArrayList.remove()这个方法看似简单,实则暗藏玄机。它提供了两种重载方式:一种是按索引删除,会返回被删除的元素;另一种是按对象内容删除,返回一个布尔值表示操作是否成功。如果选错了方法,轻则抛出IndexOutOfBoundsException,重则引发难以察觉的逻辑错误——比如你本想删除一个数字“5”,结果却把列表中第5个位置的元素给删了。问题的关键,就在于你传入的参数类型。
按索引删除:remove(int index)
当你传入一个int基本类型的下标时,触发的是这个版本。它的任务很明确:找到那个位置,移除元素,然后把后面的所有元素依次向前挪一位。这里有个硬性规定:索引必须满足0 ≤ index < size(),否则程序会毫不客气地抛出IndexOutOfBoundsException。
- 调用成功后,集合的大小(size)会减1,并且被删除位置之后的所有元素,其下标都会自动减1。
- 方法会返回被删除的那个元素,类型就是你的列表泛型E,比如String或者Integer。
- 举个例子:
list.remove(2)删除的就是列表中的第3个元素(因为索引从0开始)。
按对象删除:remove(Object o)
这个版本期待一个Object类型的参数。它的工作方式是:从列表头开始遍历,对每个元素调用equals()方法进行比较,然后删除第一个匹配成功的元素。它不关心位置,只认内容是否相等。
- 如果找到了匹配项并成功删除,返回
true;如果遍历完都没找到,则返回false。 - 需要注意两点:其一,它支持删除
null元素(使用==进行判断);其二,如果你存放的是自定义类的对象,务必确保正确重写了equals()方法。 - 例如:
list.remove("hello")会删除列表中第一个值为“hello”的字符串。
那些容易踩坑的典型场景
最常见的“坑”莫过于此:你想删除一个值为5的Integer对象,于是写了list.remove(5)。但编译器会优先将字面量5视为int类型,从而调用remove(int index),结果删除了索引为5的元素。当列表里存放的正好是Integer时,这个错误尤其隐蔽。
那么,如何规避和解决这些问题呢?
- 明确类型:如果目的是删除对象,请强制将参数转为Object类型。可以写成
list.remove((Integer)5)或者list.remove(Integer.valueOf(5))。 - 遍历时删除:如果想在循环中删除多个元素,切忌使用普通的for循环配合
remove(int),因为这会导致元素前移,从而跳过下一个待检查的元素。更安全的方式是使用Iterator.remove(),或者采用倒序的for循环。 - 删除所有匹配项:
remove(Object o)一次只能删除第一个匹配的元素。要删除所有符合条件的元素,可以使用Ja va 8引入的removeIf(Predicate)方法,或者配合迭代器进行循环删除。
立即学习“Ja va免费学习笔记(深入)”;
几个立竿见影的实用技巧
最后,分享一个快速决策的心法:
- 如果你的目标是“删除第几个”,那就用
remove(int index)。 - 如果你的目标是“删除值是多少的”,那就用
remove(Object o),并确保传入的是对象引用,而不是基本类型的字面量。 - 如果对类型不确定,又怕调错方法,可以先用
list.indexOf(o)获取对象的索引位置,再用remove(index)。这个方法在需要确认元素是否存在时特别管用。
