一、概述

  提供一种方法顺序访问一个聚合对象中各个元素, 而又无须暴露该对象的内部表示。

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

  迭代模式使用比较少,JDK集合也提供了Iterator的具体实现,可以直接拿来用,不必自己实现

1.1、适用场景

  1、访问一个聚合对象的内容而无须暴露它的内部表示。

  2、需要为聚合对象提供多种遍历方式。

  3、为遍历不同的聚合结构提供一个统一的接口。

1.2、优缺点

优点: 1、它支持以不同的方式遍历一个聚合对象。 2、迭代器简化了聚合类。 3、在同一个聚合上可以有多个遍历。 4、在迭代器模式中,增加新的聚合类和迭代器类都很方便,无须修改原有代码。

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

1.3、类图角色及其职责

   

  迭代模式的角色和职责

  1、Iterator(迭代器接口)

  该接口必须定义实现迭代功能的最小定义方法集

  比如提供hasNext()和next()方法。

  2、ConcreteIterator(迭代器实现类)

  迭代器接口Iterator的实现类。可以根据具体情况加以实现。

  3、Aggregate(容器接口)

  定义基本功能以及提供类似Iterator iterator()的方法。

  4、concreteAggregate(容器实现类)

  容器接口的实现类。必须实现Iterator iterator()方法。

1.4、演进过程

1.4.1、初始化

创建一个容器中要存储的内容Book

public class Book {
private String id;
private String name;
private double price; public Book(String id, String name, double price) {
this.id = id;
this.name = name;
this.price = price;
} public String getId() {
return id;
} public void setId(String id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public double getPrice() {
return price;
} public void setPrice(double price) {
this.price = price;
} public void display() {
System.out.println("ID=" + id + ",\tname=" + name + ",\tprice" + price);
}
}

创建一个容器BookList

public class BookList {
//容器内部还是一个List,也可以用数组
private List<Book> bookList = new ArrayList<Book>();
private int index; //添加书籍
public void addBook(Book book){
bookList.add(book);
} //删除书籍
public void removeBook(Book book){
int bookIndex = bookList.indexOf(book);
bookList.remove(bookIndex);
} //判断是否有下一本书
public boolean hasNext(){
if(index >= bookList.size()){
return false;
}
return true;
} //获得下一本书
public Book getNext(){
return bookList.get(index++);
} //获取集合长度
public int getSize(){
return bookList.size();
} //根据index获取Book
public Book getByIndex(int index){
return bookList.get(index);
}
}

  接下来,就是迭代容器了

方式一、(由容器自己实现顺序遍历。直接在容器类里直接添加顺序遍历方法)

    @Test
public void test1() {
BookList bookList = new BookList(); Book book1 = new Book("001","设计模式",200);
Book book2 = new Book("002","Java核心编程",200);
Book book3 = new Book("003","计算机组成原理",200); bookList.addBook(book1);
bookList.addBook(book2);
bookList.addBook(book3); while(bookList.hasNext()){
Book book = bookList.getNext();
book.display();
}
}

输出

ID=001,    name=设计模式,    price200.0
ID=002, name=Java核心编程, price200.0
ID=003, name=计算机组成原理, price200.0

方式二、让调用者自己实现遍历。直接暴露数据细节给外部)

    @Test
public void test2() {
BookList bookList = new BookList(); Book book1 = new Book("001", "设计模式", 200);
Book book2 = new Book("002", "Java核心编程", 200);
Book book3 = new Book("003", "计算机组成原理", 200); bookList.addBook(book1);
bookList.addBook(book2);
bookList.addBook(book3); for (int i = 0; i < bookList.getSize(); i++) {
Book book = bookList.getByIndex(i);
book.display();
}
}

结果同上

不使用迭代模式的缺点

  以上方法1与方法2都可以实现对遍历,但有这些问题

  1,容器类承担了太多功能:一方面需要提供添加删除等本身应有的功能;一方面还需要提供遍历访问功能。

  2,往往容器在实现遍历的过程中,需要保存遍历状态,当跟元素的添加删除等功能夹杂在一起,很容易引起混乱和程序运行错误等。

应用迭代模式的条件

  Iterator模式就是为了有效地处理按顺序进行遍历访问的一种设计模式,简单地说,Iterator模式提供一种有效的方法,可以屏蔽聚集对象集合的容器类的实现细节,而能对容器内包含的对象元素按顺序进行有效的遍历访问。

  所以,Iterator模式的应用场景可以归纳为满足以下几个条件:

  1、访问容器中包含的内部对象
  2、按顺序访问

1.4.2、进化

用代码实现一下迭代模式,只需修改BookList即可,BookListIterator

public class BookListIterator {
//容器内部还是一个List,也可以用数组
private List<Book> bookList = new ArrayList<Book>();
private int index; //添加书籍
public void addBook(Book book){
bookList.add(book);
} //删除书籍
public void removeBook(Book book){
int bookIndex = bookList.indexOf(book);
bookList.remove(bookIndex);
} //获取集合长度
public int getSize(){
return bookList.size();
} //根据index获取Book
public Book getByIndex(int index){
return bookList.get(index);
} //得到Iterator实例
public Iterator Iterator() {
return new Itr();
} //内部类,Iterator实例(因为要使用容器的内部信息,所以要写成内部类)
private class Itr implements Iterator{
//判断是否有下一本书,将刚才hasNext()中内容复制过来即可
public boolean hasNext() {
if(index >= bookList.size()){
return false;
}
return true;
}
//获得下一本书,将刚才getNext()中内容复制过来即可
public Object next() {
return bookList.get(index++);
} public void remove() { }
}
}

测试

    @Test
public void test3() {
BookListIterator bookList = new BookListIterator(); Book book1 = new Book("001","设计模式",200);
Book book2 = new Book("002","Java核心编程",200);
Book book3 = new Book("003","计算机组成原理",200); bookList.addBook(book1);
bookList.addBook(book2);
bookList.addBook(book3); Iterator iterator = bookList.Iterator();
while(iterator.hasNext()){
Book book = (Book) iterator.next();
book.display();
}
}

输出同上

  可以看到,这和使用JDK提供集合的Iterator方法就一模一样了。

二、扩展

2.1 JDK

java.util.Iterator
java.util.ArrayList中的Itr

2.2 Mybatis

org.apache.ibatis.cursor.defaults.DefaultCursor的cursorIterator

c

最新文章

  1. hdu4549 矩阵快速幂 + 欧拉降幂
  2. Java代理模式
  3. CodeForces 742B Batch Sort
  4. hdu4781 Assignment For Princess(构造)
  5. 气泡形提示控件grumble.js
  6. 使用repeater控件显示列表替代treeview
  7. POJ1094 Sorting It All Out(拓扑排序)
  8. CSS实例:水平居中和垂直居中的多种解决方案
  9. BZOJ 1742: [Usaco2005 nov]Grazing on the Run 边跑边吃草( dp )
  10. 转 Caffe学习系列(9):运行caffe自带的两个简单例子
  11. Javaweb-request与response
  12. win10 solidity开发环境搭建
  13. ResultHandler的用法
  14. excel 导入mysql
  15. 集合04_Set
  16. Centos7 LNMP 一键安装
  17. 使用distillery&amp;&amp;docker 部署phoenix 应用
  18. Kubernetes的ConfigMap说明
  19. 理解SQL Server中索引的概念,原理
  20. 用虚拟机安装了一台Linux系统,突然想克隆一台服务器,克隆后发现无法上网,如何解决?

热门文章

  1. [牛客网 -leetcode在线编程 -02] minimum-depth-of-binary-tree -树的最短深度
  2. 原生JavaScript和jQuery的较量
  3. PHP——封装Curl请求方法支持POST | DELETE | GET | PUT 等
  4. 【测试工具】moco入门(一)
  5. 【测试用例工具】TestLinked的使用实例(转)
  6. Vue中的插槽---slot
  7. NETTY框架的使用
  8. PPT扁平化设计总结
  9. P4921 【情侣?给我烧了!】
  10. 时间模块-datetime精确到微妙级