一、介绍

迭代器模式(Iterator Pattern)又称为游标(Cursor)模式,是行为型设计模式之一。迭代器模式算是一个比较古老的设计模式,其源于对容器的访问,比如Java中的List、Map、数组等,我们知道对容器对象的访问必然会涉及遍历算法,我们可以将遍历的方法封装在容器中,或者不提供遍历方法。如果我们将遍历的方法封装到容器中,那么对于容器类来说就承担了过多的功能,容器类不仅要维护自身内部的数据元素而且还要对外提供遍历的接口方法,因为遍历状态的存储问题还不能对同一个容器同时进行多个遍历操作,如果我们不提供遍历方法而让使用者自己去实现,又会让容器的内部细节暴露无遗,正因于此,迭代模式应运而生,在客户访问类与容器体之间插入了一个第三者——迭代器,很好地解决了上面所述的弊端。

二、定义

提供一种方法顺序访问一个容器对象中的各个元素,而又不需要暴露该对象的内部表示。

三、使用场景

  • 遍历一个容器对象。

四、迭代器模式的UML类图

UML类图:

通用模式代码:

迭代器接口:

public interface Iterator<T> {/*** 是否还有下一个元素* @return true表示有,false表示没有**/boolean hasNext();/*** 返回当前位置的元素并将位置移至下一位**/T next();
}

具体迭代器类:

public class ConcreteIterator<T> implements Iterator<T>{private List<T> list;private int cursor = 0;public ConcreteIterator(List<T> list) {this.list = list;}@Overridepublic boolean hasNext() {return cursor != list.size();}@Overridepublic T next() {T obj = null;if (this.hasNext()) {obj = this.list.get(cursor++);}return obj;}
}

容器接口:

public interface Aggregation<T> {/*** 添加一个元素**/void add(T obj);/*** 移除一个元素**/void remove(T obj);/*** 获取容器的迭代器**/Iterator<T> iterator();
}

具体容器类:

public class ConcreteAggregation<T> implements Aggregation<T>{private List<T> list = new ArrayList<>();@Overridepublic void add(T obj) {list.add(obj);}@Overridepublic void remove(T obj) {list.remove(obj);}@Overridepublic Iterator<T> iterator() {return new ConcreteIterator<>(list);}
}

客户类:

public class Client {public static void main(String args[]) {Aggregation<String> a = new ConcreteAggregation<>();a.add("a");a.add("b");a.add("c");Iterator<String> iterator = a.iterator();while (iterator.hasNext()) {System.out.print(iterator.next());}}
}

角色介绍:

  • Iterator:迭代器接口,迭代器接口负责定义、访问和遍历元素的接口。

  • ConcreteIterator:具体迭代器类,具体迭代器类的目的主要是实现迭代器接口,并记录遍历的当前位置。

  • Aggregate:容器接口,容器接口负责提供创建具体迭代器角色的接口。

  • ConcreteAggregate:具体容器类,具体迭代器角色与该容器相关联。

  • Client:客户类。

五、简单实现

小民和小辉分别在公司的两个事业部,某天老板安排任务让他们俩统计一下各自部门的员工数据,这很好办嘛,建一个类用数据结构把所有员工数据存进去即可,老板要看的时候给他用for循环实现,还是比较容易的,下面就先为员工创建一个实体类:

员工实体类:

public class Employee {private String name;// 姓名private int age;// 年龄private String sex;// 性别private String position;// 职位public Employee(String name, int age, String sex, String position) {super();this.name = name;this.age = age;this.sex = sex;this.position = position;}// 简化代码,省略setter和getter方法@Overridepublic String toString() {return "Employee{" + "name='" + name + '\'' + ", age=" + age + ", sex="+ sex + ", position='" + position + '\'' + "}";}
}

小民部门:

public class CompanyMin {private List<Employee> list = new ArrayList<>();public CompanyMin(){list.add(new Employee("小民", 26, "男", "程序猿"));list.add(new Employee("小芸", 22, "女", "测试"));list.add(new Employee("小方", 18, "女", "测试"));list.add(new Employee("可儿", 21, "女", "设计"));list.add(new Employee("朗情", 19, "女", "设计")); //吐槽一下,为什么就小民一个男的,小辉部门全男的。}public List<Employee> getEmployees(){return list;}
}

小辉部门:

public class CompanyHui {private Employee[] array = new Employee[3];public CompanyHui(){array[0] = new Employee("辉哥", 28, "男", "程序猿");array[1] = new Employee("小红", 23, "男", "程序猿");array[2] = new Employee("小辉", 25, "男", "程序猿");}public Employee[] getEmployees(){return array;}
}

可见小民和小辉的内部实现是两种方式,小民的人员信息容器的内部实质是使用的一个List类存储人员信息,而小辉的实质上使用的是一个数组,如果老板要查看人员信息就必须遍历两个容器:

Boss查看:

public class Boss {public static void main(String[] args) {CompanyHui hui = new CompanyHui();Employee[] huiList = hui.getEmployees();for(int i = 0; i < huiList.length; i++){System.out.println(huiList[i]);}CompanyMin min = new CompanyMin();List minList = min.getEmployees();for(int i = 0; i < minList.size(); i++){System.out.println(minList.get(i).toString());}}
}

结果:

Employee{name='辉哥', age=28, sex=男, position='程序猿'}
Employee{name='小红', age=23, sex=男, position='程序猿'}
Employee{name='小辉', age=25, sex=男, position='程序猿'}
Employee{name='小民', age=26, sex=男, position='程序猿'}
Employee{name='小芸', age=22, sex=女, position='测试'}
Employee{name='小方', age=18, sex=女, position='测试'}
Employee{name='可儿', age=21, sex=女, position='设计'}
Employee{name='朗情', age=19, sex=女, position='设计'}

这样看似也没有问题,但是如果有多个部门,每个部门有各自的实现,那么我们就要在Boss类中增加一遍遍历逻辑,这样Boss类的功能会越来越多,同时暴露了内部细节。那么我们需要定义一个迭代器接口:

public interface Iterator {/*** 是否还有下一个元素 * * @return true表示有,false表示没有*/boolean hasNext();/*** 返回当前元素,并将位置移至下一位*/Object next();
}

小民的迭代器:

public class MinIterator implements Iterator{private List<Employee> list;private int position;public MinIterator(List<Employee> list){this.list = list;}@Overridepublic boolean hasNext() {return !(position > list.size() - 1 || list.get(position) == null);}@Overridepublic Object next() {Employee e = list.get(position);position++;return e;}}

小辉的迭代器:

public class HuiIterator implements Iterator{private Employee[] array;private int position;public HuiIterator(Employee[] array){this.array = array;}@Overridepublic boolean hasNext() {return !(position > array.length - 1 || array[position] == null);}@Overridepublic Object next() {Employee e = array[position];position++;return e;}}

定义容器类的接口:

public interface Company {/*** 返回一个迭代器对象* * @return 迭代器对象*/Iterator iterator();}

修改一下之前的两个容器类:

public class CompanyHui implements Company{private Employee[] array = new Employee[3];public CompanyHui(){array[0] = new Employee("辉哥", 28, "男", "程序猿");array[1] = new Employee("小红", 23, "男", "程序猿");array[2] = new Employee("小辉", 25, "男", "程序猿");}public Employee[] getEmployees(){return array;}@Overridepublic Iterator iterator() {return new HuiIterator(array);}
}
public class CompanyMin implements Company{private List<Employee> list = new ArrayList<>();public CompanyMin(){list.add(new Employee("小民", 26, "男", "程序猿"));list.add(new Employee("小芸", 22, "女", "测试"));list.add(new Employee("小方", 18, "女", "测试"));list.add(new Employee("可儿", 21, "女", "设计"));list.add(new Employee("朗情", 19, "女", "设计"));}public List<Employee> getEmployees(){return list;}@Overridepublic Iterator iterator() {return new MinIterator(list);}
}

Boss查看:

public class Boss {public static void main(String[] args) {CompanyHui hui = new CompanyHui();check(hui.iterator());CompanyMin min = new CompanyMin();check(min.iterator());}private static void check(Iterator iterator){while (iterator.hasNext()) {System.out.println(iterator.next().toString());}}
}

六、Android源码中的迭代器模式

1、Cursor

当我们使用SQLiteDatabase的query方法查询数据库时,会返回一个Cursor游标对象,该游标的实质就是一个具体的迭代器,我们可以使用它来遍历数据库查询所得的结果集。

七、总结

迭代器模式发展至今,几乎所有的高级语言都有相应的内置实现,对于开发者而言,已经极少会自己去实现迭代器了,所以本章内容更多的是了解而非应用。

优点:

  • 符合面向对象设计原则中的单一职责原则。

  • 支持对容器对象的多种遍历。弱化了容器类与遍历算法之间的关系。

缺点:

  • 类文件的增加。

  • 会出现ConcurrentModificationException异常。

  • 遍历过程是一个单向且不可逆的遍历。

Android设计模式之——迭代器模式相关推荐

  1. 【GOF23设计模式】迭代器模式

    [GOF23设计模式]迭代器模式 来源:http://www.bjsxt.com/  一.[GOF23设计模式]_迭代器模式.JDK内置迭代器.内部类迭代器 1 package com.test.it ...

  2. [转载] Python进阶:设计模式之迭代器模式

    参考链接: Python中的迭代器 在软件开发领域中,人们经常会用到这一个概念--"设计模式"(design pattern),它是一种针对软件设计的共性问题而提出的解决方案.在一 ...

  3. 每日学一个设计模式1——迭代器模式

    引言 精通设计模式是从码农脱颖而出的条件之一.跟着<图解设计模式>这本书学习设计模式,从今天开始,一天总结一个设计模式. 迭代器模式(一个一个遍历) 用处 隐藏遍历集合的内部结构,遍历不同 ...

  4. php迭代器实例,php设计模式之迭代器模式实例分析【星际争霸游戏案例】

    本文实例讲述了php设计模式之迭代器模式.分享给大家供大家参考,具体如下: 星际的任务关一般会有这样的设定:一开始电脑的农民不采矿,如果战斗打响,或者玩家造出第一个兵,电脑的农民开始采矿. 我们自然会 ...

  5. php foreach 循环 判断index 小于多少_PHP设计模式之迭代器模式 - 硬核项目经理

    一说到这个模式,就不得不提循环语句.在<大话设计模式>中,作者说道这个模式现在的学习意义更大于实际意义,这是为什么呢?当然就是被foreach这货给整得.任何语言都有这种类似的语法可以方便 ...

  6. 设计模式之--迭代器模式

    一.迭代器模式介绍 迭代器模式(Iterator Pattern)又称为(Cursor)模式,是行为设计模式之一.迭代器模式算是一个比较古老的设计模式,其源于对容器的访问,比如Java 中的 List ...

  7. PHP设计模式之迭代器模式

    一说到这个模式,就不得不提循环语句.在<大话设计模式>中,作者说道这个模式现在的学习意义更大于实际意义,这是为什么呢?当然就是被foreach这货给整得.任何语言都有这种类似的语法可以方便 ...

  8. 设计模式之迭代器模式(Iterator)摘录

    23种GOF设计模式一般分为三大类:创建型模式.结构型模式.行为模式. 创建型模式抽象了实例化过程,它们帮助一个系统独立于如何创建.组合和表示它的那些对象.一个类创建型模式使用继承改变被实例化的类,而 ...

  9. 设计模式复习-迭代器模式

    Iterator.H #pragma once #include <list> #include <windows.h> using namespace std;/* 设计模式 ...

最新文章

  1. [转] composer - 文档 - 命令行
  2. sublime如何实现函数折叠
  3. 团队编程项目作业5-小组评分
  4. Enterprise Architecture 13 将顺序图自动转化为协作图
  5. 【rabbitmq】Authentication failed (rejected by the remote node), please check the Erlang cookie
  6. ios 开发证书导出p12文件_开发者在上传企业签名的应用到后,做到如下几点
  7. Android平台Chromium net中的代理配置信息获取
  8. 小程序制作回到顶部按钮
  9. SaaS风暴:中国软件企业如何应对挑战?
  10. map的key可以试一个数组吗?_二维数组的 DP
  11. Intel Core Enhanced Core架构/微架构/流水线 (15) - 先进智能缓存
  12. 计算n的阶乘以及n个阶乘相加
  13. Golang实践录:命令行cobra库实例
  14. 一周信创舆情观察(8.30~9.5)
  15. 2016年国家公务员考试公告
  16. 脚本文件BAT入门(1)
  17. 期末复习—计算机组成原理—计算机的分类
  18. 不要在本该奋斗的年纪选择了安逸
  19. 计算机控制系统中a d是什么,计算机控制系统中,()代表模拟量输出。A、AIB、AOC、DID、DO - 试题答案网问答...
  20. 苹果手机3D-Touch这个功能,其实是吃鸡神器!

热门文章

  1. Oracle备份standby,Oracle 11g 利用泠备份恢复standby库
  2. 【转】Vue.js入门教程(二)在页面中引入vue的方式
  3. 【转】C#中枚举类型与静态变量
  4. 【转】如何在 Visual Studio 2019 中连接中国版 Azure
  5. 【转】Dynamics版本历史演进及Dynamics 365重要功能介绍
  6. Sharepoint学习笔记 –架构系列—10 Sharepoint的服务器端对象模型(Server Object Model) 2.内容层次结构
  7. ASP.Net请求处理机制初步探索之旅 - Part 5 ASP.Net MVC请求处理流程
  8. 深入详解JVM内存模型与JVM参数详细配置
  9. 1000并发 MySQL数据库_再送一波干货,测试2000线程并发下同时查询1000万条数据库表及索引优化...
  10. 计算机出现蓝屏怎么解决,电脑出现蓝屏故障0x00000019怎么办?