热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

设计模式笔记12:迭代器模式(IteratorPattern)






迭代器模式提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部的标示。其别名为游标(Cursor)。迭代器模式是一种对象行为型模式。


从定义可见,迭代器模式是为容器而生。很明显,对容器对象的访问必然涉及到遍历算法。你可以一股脑的将遍历方法塞到容器对象中去;或者根本不去提供什么遍历算法,让使用容器的人自己去实现去吧。这两种情况好像都能够解决问题。

然而在前一种情况,容器承受了过多的功能,它不仅要负责自己“容器”内的元素维护(添加、删除等等),而且还要提供遍历自身的接口;而且由于遍历状态保存的问题,不能对同一个容器对象同时进行多个遍历。第二种方式倒是省事,却又将容器的内部细节暴露无遗。


二、迭代器的结构

设计模式笔记12:迭代器模式(Iterator Pattern)


1) 迭代器角色(Iterator):迭代器角色负责定义访问和遍历元素的接口。
2) 具体迭代器角色(Concrete Iterator):具体迭代器角色要实现迭代器接口,并要记录遍历中的当前位置。
3) 容器角色(Container):容器角色负责提供创建具体迭代器角色的接口。
4) 具体容器角色(Concrete Container):具体容器角色实现创建具体迭代器角色的接口——这个具体迭代器角色于该容器的结构相关。



三、迭代器模式分析


  • 聚合是一个管理和组织数据对象的数据结构。

  • 聚合对象主要拥有两个职责:一是存储内部数据;二是遍历内部数据。 

  • 存储数据是聚合对象最基本的职责。

  • 将遍历聚合对象中数据的行为提取出来,封装到一个迭代器中,通过专门的迭代器来遍历聚合对象的内部数据,这就是迭代器模式的本质。迭代器模式是“单一职责原则”的完美体现。(迭代器就是将遍历内部数据的功能放在了内部实现)一个类应该只有一个引起变化的原因。根据类设计的开闭原则,我们知道要避免类内的改变,因为修改代码很容易造成许多的错误。如果有一个类具有两个改变的原因,那么这会使得将来该类的变化几率上升,而当它真的改变时,你的设计中同时有两个方面将会受到影响。

  • 迭代器模式中应用了工厂方法模式,聚合类充当工厂类,而迭代器充当产品类,由于定义了抽象层,系统的扩展性很好,在客户端可以针对抽象聚合类和抽象迭代器进行编程。

  • 由于很多编程语言的类库都已经实现了迭代器模式,因此在实际使用中我们很少自定义迭代器,只需要直接使用Java、C#等语言中已定义好的迭代器即可,迭代器已经成为我们操作聚合对象的基本工具之一。


四、迭代器模式代码示例
代码一


interface MyCollection
{
MyIterator createIterator();
}
interface MyIterator
{
void first();
void next();
boolean isLast();
Object currentItem();
}

class NewCollection implements MyCollection
{
private Object[] obj={"dog","pig","cat","monkey","pig"};
public MyIterator createIterator()
{
return new NewIterator();
}
private class NewIterator implements MyIterator
{
private int currentIndex=0;

public void first()
{
currentIndex=0;
}
public void next()
{
if(currentIndex {
currentIndex++;
}
}
public void previous()
{
if(currentIndex>0)
{
currentIndex--;
}
}
public boolean isLast()
{
return currentIndex==obj.length;
}
public boolean isFirst()
{
return currentIndex==0;
}
public Object currentItem()
{
return obj[currentIndex];
}
}
}

class Client
{
public static void process(MyCollection collection)
{
MyIterator i=collection.createIterator();
while(!i.isLast())
{
System.out.println(i.currentItem().toString());
i.next();
}
}
public static void main(String a[])
{
MyCollection collection=new NewCollection();
process(collection);
}
}
代码二、JDK中Iterator
import java.util.*;

public class IteratorDemo
{
public static void process(Collection c)
{
Iterator i=c.iterator();

while(i.hasNext())
{
System.out.println(i.next().toString());
}
}

public static void main(String args[])
{
Collection list=new HashSet();
list.add("Cat");
list.add("Dog");
list.add("Pig");
list.add("Dog");
list.add("Monkey");

process(list);
}
}
代码三、


public interface TVIterator
{
void setChannel(int i);
void next();
void previous();
boolean isLast();
Object currentChannel();
boolean isFirst();
}

public interface Television
{
TVIterator createIterator();
}

public class TCLTelevision implements Television
{
private Object[] obj={"湖南卫视","北京卫视","上海卫视","湖北卫视","黑龙江卫视"};
public TVIterator createIterator()
{
return new TCLIterator();
}

class TCLIterator implements TVIterator
{
private int currentIndex=0;

public void next()
{
if(currentIndex {
currentIndex++;
}
}

public void previous()
{
if(currentIndex>0)
{
currentIndex--;
}
}

public void setChannel(int i)
{
currentIndex=i;
}


public Object currentChannel()
{
return obj[currentIndex];
}

public boolean isLast()
{
return currentIndex==obj.length;
}

public boolean isFirst()
{
return currentIndex==0;
}
}
}

public class SkyworthTelevision implements Television
{
private Object[] obj={"CCTV-1","CCTV-2","CCTV-3","CCTV-4","CCTV-5","CCTV-6","CCTV-7","CCTV-8"};
public TVIterator createIterator()
{
return new SkyworthIterator();
}

private class SkyworthIterator implements TVIterator
{
private int currentIndex=0;

public void next()
{
if(currentIndex {
currentIndex++;
}
}

public void previous()
{
if(currentIndex>0)
{
currentIndex--;
}
}

public void setChannel(int i)
{
currentIndex=i;
}


public Object currentChannel()
{
return obj[currentIndex];
}

public boolean isLast()
{
return currentIndex==obj.length;
}

public boolean isFirst()
{
return currentIndex==0;
}
}
}

import javax.xml.parsers.*;
import org.w3c.dom.*;
import org.xml.sax.SAXException;
import java.io.*;
public class XMLUtil
{
//该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
public static Object getBean()
{
try
{
//创建文档对象
DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = dFactory.newDocumentBuilder();
Document doc;
doc = builder.parse(new File("config.xml"));

//获取包含类名的文本节点
NodeList nl = doc.getElementsByTagName("className");
Node classNode=nl.item(0).getFirstChild();
String cName=classNode.getNodeValue();

//通过类名生成实例对象并将其返回
Class c=Class.forName(cName);
Object obj=c.newInstance();
return obj;
}
catch(Exception e)
{
e.printStackTrace();
return null;
}
}
}



SkyworthTelevision

public class Client
{
public static void display(Television tv)
{
TVIterator i=tv.createIterator();
System.out.println("电视机频道:");
while(!i.isLast())
{
System.out.println(i.currentChannel().toString());
i.next();
}
}

public static void reverseDisplay(Television tv)
{
TVIterator i=tv.createIterator();
i.setChannel(5);
System.out.println("逆向遍历电视机频道:");
while(!i.isFirst())
{
i.previous();
System.out.println(i.currentChannel().toString());
}
}

public static void main(String a[])
{
Television tv;
tv=(Television)XMLUtil.getBean();
display(tv);
System.out.println("--------------------------");
reverseDisplay(tv);
}
}



五、Java迭代器
在JDK中,Iterator接口具有如下3个基本方法:

(1) Object next():通过反复调用next()方法可以逐个访问聚合中的元素。

(2) boolean hasNext():hasNext()方法用于判断聚合对象中是否还存在下一个元素,为了不抛出异常,必须在调用next()之前先调用hasNext()。如果迭代对象仍然拥有可供访问的元素,那么hasNext()返回true。

(3) void remove():用于删除上次调用next()时所返回的元素。 


Java迭代器可以理解为它工作在聚合对象的各个元素之间,每调用一次next()方法,迭代器便越过下个元素,并且返回它刚越过的那个元素的地址引用。但是,它也有一些限制,如某些迭代器只能单向移动。在使用迭代器时,访问某个元素的唯一方法就是调用next()。

六、迭代器模式优缺点
迭代器模式的优点


  • 它支持以不同的方式遍历一个聚合对象。

  • 迭代器简化了聚合类。

  • 在同一个聚合上可以有多个遍历。

  • 在迭代器模式中,增加新的聚合类和迭代器类都很方便,无须修改原有代码,满足“开闭原则”的要求。



迭代器模式的缺点


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



七、迭代器模式适用环境
在以下情况下可以使用迭代器模式:


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

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

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




八、迭代器模式应用


  • JDK 1.2 引入了新的Java聚合框架Collections。 

  • Collection是所有Java聚合类的根接口。

  • 在JDK类库中,Collection的iterator()方法返回一个java.util.Iterator类型的对象,而其子接口java.util.List的listIterator()方法返回一个java.util.ListIterator类型的对象,ListIterator是Iterator的子类。它们构成了Java语言对迭代器模式的支持,Java语言的java.util.Iterator接口就是迭代器模式的应用。 





Iterator (recognizeable
by behavioral methods sequentially returning instances of a different type
from a queue)




  • All implementations of java.util.Iterator (thus
    among others also java.util.Scanner!).



  • All implementations of java.util.Enumeration


九、参考资料



  1. 《设计模式》刘伟主编清华大学出版社

  2. 《Head First 设计模式》

  3. http://***.com/questions/1673841/examples-of-gof-design-patterns






推荐阅读
  • andr ... [详细]
  • 本文介绍了如何在C#中启动一个应用程序,并通过枚举窗口来获取其主窗口句柄。当使用Process类启动程序时,我们通常只能获得进程的句柄,而主窗口句柄可能为0。因此,我们需要使用API函数和回调机制来准确获取主窗口句柄。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 实体映射最强工具类:MapStruct真香 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • Java编程实践:深入理解方法重载
    本文介绍了Java中方法重载的概念及其应用。通过多个示例,详细讲解了如何在同一类中定义具有相同名称但不同参数列表的方法,以实现更灵活的功能调用。 ... [详细]
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • dotnet 通过 Elmish.WPF 使用 F# 编写 WPF 应用
    本文来安利大家一个有趣而且强大的库,通过F#和C#混合编程编写WPF应用,可以在WPF中使用到F#强大的数据处理能力在GitHub上完全开源Elmis ... [详细]
  • 在进行Revit插件开发时,经常会遇到窗口被其他应用程序遮挡的问题。本文将介绍如何通过简单的代码调整,确保插件窗口始终保持在Revit主界面的最前端,提升用户体验。 ... [详细]
author-avatar
过客松鼠_230
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有