leisurexi's Blog.

迭代器模式

字数统计: 1.5k阅读时长: 6 min
2019/05/26 Share

迭代器模式(Iterator Pattern),提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该对象的内部。迭代器模式属于行为型模式。

模式定义

迭代器模式就是分离了集合对象的遍历行为,抽象出一个迭代器类来负责,这样既可以做到不暴露集合的内部结构,又可让外部代码透明地访问集合内部的数据。

模式结构

首先看下结构图:

Iterator迭代器接口类

1
2
3
4
5
6
7
8
9
public interface Iterator {

//得到下一个对象
Object next();

//判断是否有下一个对象
boolean hasNext();

}

Aggregate聚集抽象类

1
2
3
4
5
6
7
8
9
public interface Aggregate {

//添加
void add(Object object);

//创建迭代器
Iterator createIterator();

}

ConcreteIterator具体迭代器类,实现Iterator

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public class ConcreteIterator implements Iterator {

private List list;
private int currentIndex = 0;

public ConcreteIterator(List list) {
this.list = list;
}

@Override
public Object next() {
if (hasNext()) {
return list.get(currentIndex++);
}
return null;
}

@Override
public boolean hasNext() {
if (currentIndex == list.size()) {
return false;
}
return true;
}

}

ConcreteAggregate集体聚集类,继承Aggregate

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class ConcreteAggregate implements Aggregate {

private List<Object> items = new ArrayList<>();

@Override
public Iterator createIterator() {
return new ConcreteIterator(items);
}

public void add(Object object) {
items.add(object);
}

}

客户端代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class Client {

public static void main(String[] args) {
Aggregate aggregate = new ConcreteAggregate();
aggregate.add(1);
aggregate.add(2);
aggregate.add(3);
aggregate.add(4);

Iterator iterator = aggregate.createIterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}

}

}

运行结果

1
2
3
4
1
2
3
4

模式实现

在java中有一个Iterable接口,实现该接口的类就是可迭代的,并且支持增强for循环。该接口只有一个方法即获取迭代器的方法iterator(),可以获取每个容易自身的迭代器Iterator。

1
2
3
4
public interface Iterable<T> {
//返回迭代器对象
Iterator<T> iterator();
}

Iterator接口中的方法。

1
2
3
4
5
6
7
8
9
10
public interface Iterator<E> {
//是否有下一个元素
boolean hasNext();
//下一个元素
E next();
//将删除上次调用next方法时返回的元素,如果想要删除指定位置上的元素,需要越过这个元素
//next方法和remove方法是相互依赖的,如果调用remove方法之前没有调用next将是不合法的,
//将会抛出IllegalStateException异常
void remove()
}

Collection继承了Iterable接口,所以Collection体系都具备获取自身迭代器的方法,只不过每个子类集合都进行了重写(因为数据结构不同)。

接下来我们主要看下ArrayList中是怎么实现的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
   //调用ArrayList的该方法返回内部实现Iterator接口的Itr对象
public Iterator<E> iterator() {
return new Itr();
}

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;

Itr() {}

public boolean hasNext() {
//如果下一个元素的下表不等于容器中元素的实际大小,返回true代表有下一个元素
return cursor != size;
}

@SuppressWarnings("unchecked")
public E next() {
//检查被迭代的对象是否被修改过
checkForComodification();
int i = cursor;
//如果下标大于等于容器中元素的实际大小抛出异常
if (i >= size)
throw new NoSuchElementException();
Object[] elementData = ArrayList.this.elementData;
//如果下标大于等于列表长度,抛出异常
if (i >= elementData.length)
throw new ConcurrentModificationException();
//更新下个元素的下标
cursor = i + 1;
//返回列表中当前下标的元素,并对lastRet赋值
return (E) elementData[lastRet = i];
}

public void remove() {
//如果lastRet小于0代表没调用过next()方法,抛出异常
if (lastRet < 0)
throw new IllegalStateException();
//检查被迭代的对象是否被修改过
checkForComodification();

try {
//根据调用next()给lastRet赋值的下标去删除元素
ArrayList.this.remove(lastRet);
//因为删除元素后容器内元素实际大小减1,给cursor赋值未加1前的下标
cursor = lastRet;
//lastRet重置
lastRet = -1;
//更新expectedModCount的值
expectedModCount = modCount;
} catch (IndexOutOfBoundsException ex) {
throw new ConcurrentModificationException();
}
}

final void checkForComodification() {
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
}
}

checkForComodification()方法检查所迭代的列表对象是否被修改过,modCount是外部类的一个字段,当调用外部类的add,remove,ensureCapacity等方法,都会改变该字段的值,而expectedModCount是内部类Itr初始化时,使用modCount赋值的字段。这样,在使用迭代器过程中,如果正在对正在迭代的对象调用了add,remvoe,ensureCapacity等方法,再去调用迭代器的next方法,就会引发ConcurrentModificationException异常了。

代码示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test
public void test2() {
List list = new ArrayList();
list.add(1);
list.add(2);
list.add(3);

Iterator iterator = list.iterator();
//while遍历
while (iterator.hasNext()) {
System.out.println(iterator.next());

}
}

运行结果

1
2
3
1
2
3

如果如上面所说在循环过程中修改list的列表对象,看是否会引发异常。我们队上面代码简单修改一下。

1
2
3
4
5
6
7
8
9
10
11
12
13
@Test
public void test2() {
List list = new ArrayList();
list.add(1);
list.add(2);
list.add(3);

Iterator iterator = list.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
list.add(6);
}
}
1
2
3
4
5
6
1

java.util.ConcurrentModificationException
at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
at java.util.ArrayList$Itr.next(ArrayList.java:859)
at com.lwx.arraylist.ArrayListTest.test2(ArrayListTest.java:40)

可以看出在遍历第二个元素调用next()方法时,就会引发异常。

模式的优缺点

优点

  • 在迭代器模式中,增加新的聚合类和迭代器类都很方便,无须修改原有代码。
  • 它支持以不同的方式遍历一个聚合对象。

缺点

  • 由于迭代器模式将存储数据和遍历数据的职责分离,增加新的聚合类需要对应增加新的迭代器,类的个数成对增加,这在一定程度上增加了系统的复杂性。

模式使用场景

当你需要对聚集有多种方式遍历时,可以考虑用迭代器模式。不过大多数高级语言都对它进行了封装,所以给人感觉这种模式本身不太常用了。

CATALOG
  1. 1. 模式定义
  2. 2. 模式结构
  3. 3. 模式实现
    1. 3.1. 代码示例
  4. 4. 模式的优缺点
  5. 5. 模式使用场景