关于ArrayList的文章有两篇,前一篇写了ArrayList的数据结构、扩容机制、源码分析等,这篇来看一下ArrayList的一些操作细节。
1. 删除元素操作
1.1 通过foreach删除
List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("3");
for (String element : list) {
System.out.println(element);
list.remove(element);//删除操作
}
使用foreach
遍历数组,其实底层使用的Iterator
,在Iterator
创建的时候,维护了一个全局变量expectedModCount
,将ArrayList
的modCount
赋值给expectedModCount
。当list发生增、删、改的时候modCount
会做递增,但是expectedModCount
不会变。
Iterator
在执行next
方法的时候获取检测expectedModCount
和modCount
是否相等,当不相等的时候会抛出ConcurrentModificationException
异常。由此可见,上面的代码在执行的过程中或出现异常的问题。
解释源码如下:
private class Itr implements Iterator<E> {
int cursor; // index of next element to return
int lastRet = -1; // index of last element returned; -1 if no such
int expectedModCount = modCount; //初始化expectedModCount,与modCount等
//……
}
public E next() {
checkForComodification();// 检查expectedModCount、modCount
int i = cursor;
if (i >= size)
throw new NoSuchElementException();
Object[] elementData = ArrayList.this.elementData;
if (i >= elementData.length)
throw new ConcurrentModificationException();
cursor = i + 1;
return (E) elementData[lastRet = i];
}
// 检查expectedModCount、modCount
final void checkForComodification() {
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
}
1.2 通过for循环删除
List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("3");
for (int i = 0; i < list.size(); i++) {
System.out.println(list.get(i));
list.remove(i);
}
此方式循环删除,不会出现异常问题,因为使用的是ArrayList本身的API,不会触发Iterator
的检查机制。但是会出现数据遍历不全的问题。我们都知道在ArrayList中删除元素后,被删除的元素后面的元素会向前移动一位,这个时候就会导致被删除元素的下一个元素移动到删除元素的位置,出现遗漏遍历的问题,具体如下:
- 当遍历角标为0的元素后,删除角标为0的元素,此时原来的数组就会变成第二种情况,所有元素向前移动一位;
- 当循环循环继续,此时遍历的角标为1的元素,取到的值不是2,而是3,也就是说2漏掉了,没有遍历到。
1.3 正确的ArrayList增删改姿势
通过上面的两种删除操作比较,都是存在很大的问题的,在实际开发都不能使用,但是此时就需要对ArrayList遍历,并进行增删改操作怎么办呢?那就通过Iterator
或者ListIterator
来实现。
List<String> list = new ArrayList<>();
list.add("1");
list.add("2");
list.add("2");
list.add("3");
System.out.println("删除前的List:" + list);
ListIterator<String> listIterator = list.listIterator();
// 或者直接使用lambda表达式代替整个while循环:list.removeIf("2"::equals); -- 一行代码搞定
while (listIterator.hasNext()) {
String element = listIterator.next();
if ("2".equals(element)) { //删除list集合值为“2”的元素
listIterator.remove();
}
}
System.out.println("删除后的List:" + list);
为什么这样就可以了呢?不会出现ConcurrentModificationException
异常吗?下面看一下源码解释!
// 这个对应的是Iterator内部的add方法
public void add(E e) {
checkForComodification(); // 检查expectedModCount、modCount
try {
int i = cursor;
ArrayList.this.add(i, e); // 添加元素
cursor = i + 1;
lastRet = -1;
expectedModCount = modCount; // 重新给expectedModCount赋值
} catch (IndexOutOfBoundsException ex) {
throw new ConcurrentModificationException();
}
}
// 这个对应的是Iterator内部的remove方法
public void remove() {
if (lastRet < 0)
throw new IllegalStateException();
checkForComodification();// 检查expectedModCount、modCount
try {
SubList.this.remove(lastRet);// 删除元素
cursor = lastRet;
lastRet = -1;
expectedModCount = ArrayList.this.modCount;//重新给expectedModCount赋值
} catch (IndexOutOfBoundsException ex) {
throw new ConcurrentModificationException();
}
}
从上面的代码可以看出来,在通过Iterator
(或者ListIterator
)增、删方法操作后会重新维护expectedModCount
的值,保证与ArrayList.this.modCount
相同,再下次进行checkForComodification()
方法时,自然不会出现两者值不同的问题,就不会出现ConcurrentModificationException
异常。
2. ArrayList扩容和移位的区别
ArrayList扩容是对动态数组做增长操作,而移位是在原有的数组长度不变的情况下,移动部分元素在数组中的位置。举个例子:
- 开会的时候,有一排5个座位,此时坐了4个人,挨着坐的,后来来了一个领导,就需要将中间位置让给领导,那么就需要其中两位移动一下位置,坐变边上的两个座位,领导插入中间。这就是移位,本身座位个数是没有变化的。
- 一个盒子只能装下4个苹果,现在需要再放一个苹果,那就要重新寻找一个大盒子,把之前小盒子里面的苹果以及增加的这个苹果一起放到大盒子中。这个就是扩容了,存储苹果的盒子及位置发生了变化。
细心的同学在看ArrayList
源码的时候可以看到,其内部频繁使用了两个方法System.arraycopy()
和Arrays.copyOf()
。
2.1 扩容Arrays.copyOf()
此方法创建一个新的数组,将原来的数组复制到新的数组中。此时数组在内存的地址发生了变化。也就是说,在扩容的时候,是在内存中重新开辟一个连续的空间,然后将原数组的数据复制进去。
2.2 移位System.arraycopy()
此方法是将原数组中的部分复制到原数组的另一个位置,空出位置留给插入元素使用,原数组地址不变。也就是说,此操作是在原数组内部发生的,不会涉及内存空间的重新开辟。