迭代器模式屬于行為型模式,其意圖是提供一種方法順序訪問一個聚合對象中得各個元素,而又不需要暴露該對象的內部表示。一個聚合對象,比如列表,應該提供一種方法來讓別人可以訪問他的元素。而又不需要暴露他的內部結構,此外,針對不同的需求,可能要以以下不同的方式遍歷這個列表,但是即使可以預見所需要的那些遍歷操作,你可能也不希望列表的接口中充斥著各種不同的遍歷操作。有時還可能需要在同一個列表上同時進行多個遍歷。迭代器模式可以幫助你解決所有這些問題。這一個模式的關鍵思想市將對列表的訪問和遍歷從列表對象中分離出來并放入一個迭代器對象中,迭代器類定義了一個訪問該列表元素的接口,迭代器對象負責跟蹤當前的元素,即他知道那些元素已經遍歷過了。在實例化列表迭代器之前,必須提供待遍歷的列表,一旦有了該列表迭代器的實例,就可以順序訪問該列表的各個元素。CurrentItem操作返回列表中得當前元素,First操作初始化迭代器,使當前元素指向列表的第一個元素,Next操作將當前元素指針向前推進一步,指向下一個元素,而IsDone檢查是否已經越過最后一個元素,也就是完成了這次遍歷。
例如常見的電視,可以定義一個 遙控器的,每個工廠生產的電視機都有配套的遙控器,這有點像抽象工廠模式,但是現在不是說抽象工廠模式,而說的重點是迭代器模式,電視的頻道為Item,每個遙控器都會有遍歷的功能,用以遍歷所有的頻道。如下圖所示,
適用性:
l 訪問一個聚合對象的內容而無需暴露他的內部表示。
l 支持對句和對象的多遍遍歷。
l 為遍歷不同的聚合結構提供一個統一的接口。
參與者:
Iterator:迭代器定義訪問和遍歷元素的接口。
ConcreteIterator(Controller):具體迭代器實現迭代器接口,對該聚合遍歷時跟蹤當前位置。
Aggregate(Televation):聚合定義了創建相應迭代器對象的接口。
ConcreteAggregate(HaierTV):具體聚合實現創建相應迭代器的接口,該操作返回ConcreteIterator的一個適當的實例。
協作關系:ConcreteIterator跟蹤聚合中的當前對象,并能夠計算出待遍歷的后繼對象。
使用迭代器的好處:
1. 他支持以不同的方式遍歷一個聚合, 復雜的聚合可用多種方式進行遍歷。
2. 迭代器簡化了聚合的接口 有了迭代器的遍歷接口,聚合本身就不需要類似的遍歷接口了,這樣就簡化了聚合的接口。
3. 在同一個聚合上可以有多個遍歷 每個迭代器保持它自己的遍歷狀態。因此你可以同時進行多個遍歷。
在本例子中,Television定義了一個返回各個頻道列表的接口,這實際上是一個工廠方法,只是生產出來的產品所屬的類型支持Iterator的操作。
具體的代碼如下所示:
Iterator接口:
package iterator;
public interface Iterator{
public Item first();
public Item next();
public boolean isDone();
public Item currentItem();
}
Controller類實現了Iterator接口。
package iterator;
import java.util.Vector;
public class Controller implements Iterator{
private int current =0;
Vector channel;
public Controller(Vector v){
channel = v;
}
public Item first(){
current = 0;
return (Item)channel.get(current);
}
public Item next(){
current ++;
return (Item)channel.get(current);
}
public Item currentItem(){
return (Item)channel.get(current);
}
public boolean isDone(){
return current>= channel.size()-1;
}
}
Television接口:
package iterator;
import java.util.Vector;
public interface Television{
public Iterator createIterator();
public Vector getChannel();
}
HaierTV類實現了Television接口。
package iterator;
import java.util.Vector;
public class HaierTV implements Television{
private Vector channel;
public HaierTV(){
channel = new Vector();
channel.addElement(new Item("channel 1"));
channel.addElement(new Item("channel 2"));
channel.addElement(new Item("channel 3"));
channel.addElement(new Item("channel 4"));
channel.addElement(new Item("channel 5"));
channel.addElement(new Item("channel 6"));
channel.addElement(new Item("channel 7"));
}
public Vector getChannel(){
return channel;
}
public Iterator createIterator(){
return new Controller(channel);
}
}
Client客戶端:
package iterator;
public class Client{
public static void main(String[] args){
Television tv = new HaierTV();
Iterator it =tv.createIterator();
System.out.println(it.first().getName());
while(!it.isDone()){
System.out.println(it.next().getName());
}
}
}
Item類的接口:
package iterator;
public class Item{
private String name;
public Item(String aName){
name = aName;
}
public String getName(){
return name;
}
}
總結:Iterator模式提供了一個訪問聚合數據的簡單思路,并支持多個迭代器同時訪問。