一、 迭代器(Iterator)模式

迭代器是针对集合对象而生的,对于集合对象而言,必然涉及到集合元素的添加删除操作,同时也肯定支持遍历集合元素的操作,我们此时可以把遍历操作也放在集合对象中,但这样的话,集合对象就承担太多的责任了,面向对象设计原则中有一条是单一职责原则,所以我们要尽可能地分离这些职责,用不同的类去承担不同的职责。迭代器模式就是用迭代器类来承担遍历集合元素的职责。

迭代器模式提供了一种方法顺序访问一个聚合对象(理解为集合对象)中各个元素,而又无需暴露该对象的内部表示,这样既可以做到不暴露集合的内部结构,又可让外部代码透明地访问集合内部的数据。

二、 迭代器模式的结构

既然,迭代器模式承担了遍历集合对象的职责,则该模式自然存在2个类,一个是聚合类,一个是迭代器类。在面向对象涉及原则中还有一条是针对接口编程,所以,在迭代器模式中,抽象了2个接口,一个是聚合接口,另一个是迭代器接口,这样迭代器模式中就四个角色了,具体的类图如下所示:

从上图可以看出,迭代器模式由以下角色组成:

  • 迭代器角色(Iterator):迭代器角色负责定义访问和遍历元素的接口
  • 具体迭代器角色(Concrete Iteraror):具体迭代器角色实现了迭代器接口,并需要记录遍历中的当前位置。
  • 聚合角色(Aggregate):聚合角色负责定义获得迭代器角色的接口
  • 具体聚合角色(Concrete Aggregate):具体聚合角色实现聚合角色接口。

三、 迭代器模式的实现

  1. // 抽象聚合类
  2.  
  3. using System;
  4.  
  5. public interface IListCollection
  6. {
  7. Iterator GetIterator();
  8. }
  9.  
  10. // 迭代器抽象类
  11. public interface Iterator
  12. {
  13. bool MoveNext();
  14. Object GetCurrent();
  15. void Next();
  16. void Reset();
  17. }
  18.  
  19. // 具体聚合类
  20. public class ConcreteList : IListCollection
  21. {
  22. readonly int[] _collection;
  23. public ConcreteList()
  24. {
  25. _collection = new int[] { , , , };
  26. }
  27.  
  28. public Iterator GetIterator()
  29. {
  30. return new ConcreteIterator(this);
  31. }
  32.  
  33. public int Length
  34. {
  35. get { return _collection.Length; }
  36. }
  37.  
  38. public int GetElement(int index)
  39. {
  40. return _collection[index];
  41. }
  42. }
  43.  
  44. // 具体迭代器类
  45. public class ConcreteIterator : Iterator
  46. {
  47. // 迭代器要集合对象进行遍历操作,自然就需要引用集合对象
  48. private readonly ConcreteList _list;
  49. private int _index;
  50.  
  51. public ConcreteIterator(ConcreteList list)
  52. {
  53. _list = list;
  54. _index = ;
  55. }
  56.  
  57. public bool MoveNext()
  58. {
  59. if (_index < _list.Length)
  60. {
  61. return true;
  62. }
  63. return false;
  64. }
  65.  
  66. public Object GetCurrent()
  67. {
  68. return _list.GetElement(_index);
  69. }
  70.  
  71. public void Reset()
  72. {
  73. _index = ;
  74. }
  75.  
  76. public void Next()
  77. {
  78. if (_index < _list.Length)
  79. {
  80. _index++;
  81. }
  82.  
  83. }
  84. }
  85.  
  86. // 客户端
  87. class Program
  88. {
  89. static void Main(string[] args)
  90. {
  91. IListCollection list = new ConcreteList();
  92. Iterator iterator = list.GetIterator();
  93.  
  94. while (iterator.MoveNext())
  95. {
  96. var i = (int)iterator.GetCurrent();
  97. Console.WriteLine(i.ToString());
  98. iterator.Next();
  99. }
  100.  
  101. Console.Read();
  102. }
  103. }

自然,上面代码的运行结果也是对集合每个元素的输出,具体运行结果如下所示:

  1. 2
  2. 4
  3. 6
  4. 8

四、 .NET中迭代器模式的应用

在.NET下,迭代器模式中的聚集接口和迭代器接口都已经存在了,其中IEnumerator接口扮演的就是迭代器角色,IEnumberable接口则扮演的就是抽象聚集的角色,只有一个GetEnumerator()方法,关于这两个接口的定义可以自行参考MSDN。在.NET 1.0中,.NET 类库中很多集合都已经实现了迭代器模式,大家可以用反编译工具Reflector来查看下mscorlib程序集下的System.Collections命名空间下的类,这里给出ArrayList的定义代码,具体实现代码可以自行用反编译工具查看,具体代码如下所示:

  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4.  
  5. public class ArrayList : IList, ICollection, IEnumerable, ICloneable
  6. {
  7. // Fields
  8. private const int DefaultCapacity = ;
  9. private object[] _items;
  10. private int _size;
  11. [NonSerialized]
  12. private object _syncRoot;
  13. private int _version;
  14.  
  15. public ArrayList(object[] items, int size, object syncRoot, int version)
  16. {
  17. _items = items;
  18. _size = size;
  19. _syncRoot = syncRoot;
  20. _version = version;
  21. }
  22.  
  23. public virtual IEnumerator GetEnumerator();
  24. public virtual IEnumerator<> GetEnumerator(int index, int count);
  25.  
  26. // Properties
  27. public virtual int Capacity { get; set; }
  28. public virtual int Count { get; }
  29. ..............// 更多代码请自行用反编译工具Reflector查看
  30. }

通过查看源码你可以发现,ArrayList中迭代器的实现与我们前面给出的示例代码非常相似。然而,在.NET 2.0中,由于有了yield return关键字,实现迭代器模式就更简单了。

五、 迭代器模式的适用场景

在下面的情况下可以考虑使用迭代器模式:

  1. 系统需要访问一个聚合对象的内容而无需暴露它的内部表示。
  2. 系统需要支持对聚合对象的多种遍历。
  3. 系统需要为不同的聚合结构提供一个统一的接口。

六、 迭代器模式的优缺点

优点:

  1. 迭代器模式使得访问一个聚合对象的内容而无需暴露它的内部表示,即迭代抽象。
  2. 迭代器模式为遍历不同的集合结构提供了一个统一的接口,从而支持同样的算法在不同的集合结构上进行操作

缺点:

  1. 迭代器模式在遍历的同时更改迭代器所在的集合结构会导致出现异常。所以使用foreach语句只能在对集合进行遍历,不能在遍历的同时更改集合中的元素。

七、 总结

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

C#设计模式-迭代器模式的更多相关文章

  1. 19. 星际争霸之php设计模式--迭代器模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  2. java设计模式——迭代器模式

    一. 定义与类型 定义:提供一种方法,顺序访问一个集合对象中的各个元素,而又不暴露该对象的内部表示 类型:行为型. 二. 使用场景 (1) 访问一个集合对象的内容而无需暴露它的内部表示 (2)  为遍 ...

  3. [Head First设计模式]生活中学设计模式——迭代器模式

    系列文章 [Head First设计模式]山西面馆中的设计模式——装饰者模式 [Head First设计模式]山西面馆中的设计模式——观察者模式 [Head First设计模式]山西面馆中的设计模式— ...

  4. JAVA 设计模式 迭代器模式

    用途 迭代器模式 (Iterator) 提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该对象的内部表示. 迭代器模式是一种行为型模式. 结构

  5. 深入浅出设计模式——迭代器模式(Iterator Pattern)

    模式动机 一个聚合对象,如一个列表(List)或者一个集合(Set),应该提供一种方法来让别人可以访问它的元素,而又不需要暴露它的内部结构.针对不同的需要,可能还要以不同的方式遍历整个聚合对象,但是我 ...

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

    迭代器模式是指提供一种方法顺序访问一个聚合对象中的各个元素,而又不需要暴露该对象的内部表示. 迭代器模式可以把迭代的过程从业务逻辑中分离出来,在使用迭代器模式之后,即使不关心对象的内部构造,也可以按顺 ...

  7. javascript设计模式-迭代器模式(Iterator)

    <!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. 设计模式 --迭代器模式(Iterator)

    能够游走于聚合内的每一个元素,同时还可以提供多种不同的遍历方式.   基本概念: 就是提供一种方法顺序访问一个聚合对象中的各个元素,而不是暴露其内部的表示.   使用迭代器模式的优点: 遍历集合或者数 ...

  9. js设计模式--迭代器模式

    迭代器模式: 迭代器模式提供一种方法顺序访问一个聚合对象中各个元素,而又不需要暴露该方法中的内部表示.js中我们经常会封装一个each函数用来实现迭代器. 理解的意思:提供一个方法,去把对象的每一项按 ...

  10. javascript设计模式——迭代器模式

    前面的话 迭代器模式是指提供一种方法顺序访问一个聚合对象中的各个元素,而又不需要暴露该对象的内部表示.迭代器模式可以把迭代的过程从业务逻辑中分离出来,在使用迭代器模式之后,即使不关心对象的内部构造,也 ...

随机推荐

  1. C# 在腾讯的发展

    本文首发我的微信公众号"dotnet跨平台", 内容得到大家热烈的欢迎,全文重新发布在博客,欢迎转载,请注明出处. .NET 主要的开发语言是 C# , .NET 平台泛指遵循EC ...

  2. Swift3.0服务端开发(一) 完整示例概述及Perfect环境搭建与配置(服务端+iOS端)

    本篇博客算是一个开头,接下来会持续更新使用Swift3.0开发服务端相关的博客.当然,我们使用目前使用Swift开发服务端较为成熟的框架Perfect来实现.Perfect框架是加拿大一个创业团队开发 ...

  3. myeclipse学习总结一(在MyEclipse中设置生成jsp页面时默认编码为utf-8编码)

    1.每次我们在MyEclispe中创建Jsp页面,生成的Jsp页面的默认编码是"ISO-8859-1".在这种情况下,当我们在页面中编写的内容存在中文的时候,就无法进行保存.如下图 ...

  4. 06.SQLServer性能优化之---数据库级日记监控

    汇总篇:http://www.cnblogs.com/dunitian/p/4822808.html#tsql 之前说了一下数据库怎么发邮件:http://www.cnblogs.com/duniti ...

  5. 【原】AFNetworking源码阅读(三)

    [原]AFNetworking源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇的话,主要是讲了如何通过构建一个request来生成一个data tas ...

  6. H5坦克大战之【玩家控制坦克移动】

    自从威少砍下45+11+11的大号三双之后,网上出现了各种各样的神级段子,有一条是这样的: 威少:Hey,哥们,最近过得咋样! 浓眉:对方开启了好友验证,请先添加对方为好友 威少:...... JRS ...

  7. [转载]一个标准java程序员的进阶过程

    第一阶段:Java程序员 技术名称 内                 容 说明 Java语法基础 基本语法.数组.类.继承.多态.抽象类.接口.object对象.常用类(Math\Arrarys\S ...

  8. Html 制作相册

    本文主要讲述采用Html5+jQuery+CSS 制作相册的小小记录. 主要功能点: Html5进行布局 调用jQuery(借用官网的一句话:The Write Less, Do More)极大的简化 ...

  9. MySQL ALTER命令

    当我们需要修改数据表名或者修改数据表字段时,就需要使用到MySQL ALTER命令. 开始本文教程前让我们先创建一张表,表名为:testalter_tbl. root@host# mysql -u r ...

  10. [jquery]显示隐藏div标签的几种方法

    1.$("#demo").attr("style","display:none;");//隐藏div $("#demo" ...