迭代器模式

迭代器模式

介绍

迭代器模式是一种设计模式,它提供一种方法访问一个容器对象中各个元素,而不需要暴露该对象的内部表示。

迭代器模式的主要目的是:

  1. 提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部的表示。
  2. 为遍历不同的集合结构提供一个统一的接口(即迭代器接口)。

迭代器模式的主要 PARTICIPANT 包括:

Iterator 抽象迭代器接口:定义访问和遍历元素的方法。

ConcreteIterator 具体迭代器实现:实现抽象迭代器接口中定义的方法。

Aggregate 抽象容器接口:定义创建相应迭代器对象的方法。

ConcreteAggregate 具体容器实现:实现创建具体迭代器的方法。

客户端可以通过迭代器来遍历容器中的所有元素,而不需要了解容器的内部结构。迭代器简化了聚合类的接口和聚合类的遍历。

迭代器模式在提供一个统一遍历接口的同时,封装了集合的内部结构,使客户端代码与集合的存储结构解耦。

定义

假设我们有一个图书管理系统,需要管理图书馆中的所有图书。图书馆有打印版图书和电子版图书,打印版图书存储在书架上,电子版存储在服务器上。

我们希望提供一个统一的界面来遍历所有图书,而不管是打印版还是电子版。

需求如下:

  1. 图书馆有一个所有图书的列表,包括打印版和电子版图书。
  2. 图书馆可以提供一个迭代器接口,用来遍历所有图书。
  3. 打印版图书存储在书架上,需要提供具体的书架迭代器来遍历书架上的图书。
  4. 电子版图书存储在服务器上,需要提供具体的服务器迭代器来遍历服务器上的电子图书。
  5. 使用图书馆提供的迭代器接口,可以一致地遍历所有图书,而不需要关心图书的存储形式是打印版还是电子版。

图书接口

// Book接口
class Book {
public:
  virtual string getName() = 0;
};
// 具体书类
class DesignPatternsBook : public Book {
public:
  string getName() override ;
};
// 具体书类 
class RefactoringBook : public Book {
public:
   string getName() override ;
};

具体实现

// 具体书类
string DesignPatternsBook::getName() { return "Design Patterns"; }
// 具体书类
string RefactoringBook::getName() { return "Refactoring"; }

图书聚合类

// BookShelf类,实现Aggregate接口
class BookShelf {
private:
  vector<Book*> books;
public:
  vector<Book*>& Books() ;
  void addBook(Book* book) ;
  // 创建迭代器
  Iterator* createIterator() ;
};

实现

vector<Book*>& BookShelf::Books() { return books; }
void BookShelf::addBook(Book* book) { books.push_back(book); }
// 创建迭代器
Iterator* BookShelf::createIterator() { return new BookShelfIterator(this); }

迭代器接口

// 迭代器接口
class Iterator {
public:
  virtual bool hasNext() = 0;
  virtual Book* next() = 0;
};
// 具体迭代器
class BookShelfIterator : public Iterator {
private:
  BookShelf* bookShelf;
  int index;
public:
  BookShelfIterator(BookShelf* bookShelf) ;
  bool hasNext() override ;
  Book* next() override ;
};

实现

// 具体迭代器
BookShelfIterator::BookShelfIterator(BookShelf* bookShelf) {
  this->bookShelf = bookShelf;
  this->index = 0;
}
bool BookShelfIterator::hasNext() {
  if (index < bookShelf->Books().size()) {
    return true;
  }
  return false;
}
Book* BookShelfIterator::next() {
  Book* book = bookShelf->Books()[index];
  index++;
  return book;
}

调用

// 客户端代码
int main() {
  BookShelf* bookShelf = new BookShelf();
  bookShelf->addBook(new DesignPatternsBook());
  bookShelf->addBook(new RefactoringBook());
  Iterator* it = bookShelf->createIterator();
  while (it->hasNext()) {
    Book* book = it->next();
    cout << book->getName() << endl; 
  }
  return 0;
}

效果

./bin/design/iterator 
Design Patterns
Refactoring

回顾

迭代器模式,也叫游标模式。它用来遍历集合对象。这里说的“集合对象”,我们也可以叫“容器”“聚合对象”,实际上就是包含一组对象的对象,比如,数组、链表、树、图、跳表。

一个完整的迭代器模式,一般会涉及容器和容器迭代器两部分内容。为了达到基于接口而非实现编程的目的,容器又包含容器接口、容器实现类,迭代器又包含迭代器接口、迭代器实现类。容器中需要定义iterator()方法,用来创建迭代器。

遍历集合一般有三种方式:for循环、迭代器遍历。相对于for循环遍历,利用迭代器来遍历有下面三个优势:

  • 迭代器模式封装集合内部的复杂数据结构,开发者不需要了解如何遍历,直接使用容器提供的迭代器即可;
  • 迭代器模式将集合对象的遍历操作从集合类中拆分出来,放到迭代器类中,让两者的职责更加单一;
  • 迭代器模式让添加新的遍历算法更加容易,更符合开闭原则。除此之外,因为迭代器都实现自相同的接口,在开发中,基于接口而非实现编程,替换迭代器也变得更加容易。
    代码
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值