C# 迭代器 Iterator

背景

由于枚举器存在遍历二叉树不方便的问题。才有了迭代器。在了解了迭代器之前必须了解枚举器。

迭代器就是带了状态机的枚举器。

基本介绍

1)迭代器模式( lterator Pattern)是常用的设计模式,属于行为型模式
2)如果我们的集合元素是用不同的方式实现的,有数组,还有java的集合类,
或者还有其他方式,当客户端要遍历这些集合元素的时候就要使用多种遍历方式,而且还会暴露元素的内部结构,可以考虑使用迭代器模式解决。
3)迭代器模式,提供一种遍历集合元素的统一接口,用一致的方法遍历集合元素,
不需要知道集合对象的底层表示,即:不暴露其内部的结构。

迭代器简介

至此,你已了解 foreach 的内部实现代码,是时候了解如何使用迭代器创建 IEnumerator<T>、IEnumerable<T> 和自定义集合对应的非泛型接口的自定义实现代码了。迭代器提供明确的语法,用于指定如何迭代集合类中的数据,尤其是使用 foreach 循环。这样一来,集合的最终用户就可以浏览其内部结构,而无需知道相应结构。

枚举模式存在的问题是,手动实现起来不方便,因为必须始终指示描述集合中的当前位置所需的全部状态。对于列表集合类型类,指示这种内部状态可能比较简单;当前位置的索引就足够了。相比之下,对于需要递归遍历的数据结构(如二叉树),指示状态可能就会变得相当复杂。为了减少实现此模式所带来的挑战,C# 2.0 新增了 yield 上下文关键字,这样类就可以更轻松地决定 foreach 循环如何迭代其内容。

 

语法信息

1、迭代器可用作一种方法,或一个 get 访问器。 不能在事件、实例构造函数、静态构造函数或静态终结器中使用迭代器。
2、必须存在从 yield return 语句中的表达式类型到迭代器返回的 IEnumerable<T> 类型参数的隐式转换。
3、在 C# 中,迭代器方法不能有任何 in、ref 或 out 参数。
4、在 C# 中,yield 不是保留字,只有在 return 或 break 关键字之前使用时才有特殊含义。

迭代器块

迭代器块是有一个或多个yield语句的代码块。下面3种类型的代码块中的任意一种都可以是迭代器块:

  • 方法主体
  • 访问器主体
  • 运算符主体

迭代器块与其他代码块不同。其他块包含的语句被当做命令式。即先执行代码块中的第一个语句,然后执行后面的语句,最后控制离开块。
另一方面,迭代器块不是需要在同一时间执行的一串命令式命令,而是描述了希望编译器为我们创建的枚举器类的行为。迭代器块中的代码描述了如何枚举元素。
迭代器块由两个特殊语句:

  • yield return语句指定了序列中返回的下一项
  • yield break语句指定在序列中没有的其他项

编译器得到有关枚举项的描述后,使用它来构建包含所有需要的方法和属性实现的枚举器类。结果类被嵌套包含在迭代器声明的类中。
如下图所示,根据迭代器块的返回类型,你可以让迭代器产生枚举器或可枚举类型。 

 

 

使用迭代器来创建枚举器

class MyClass
{
    //该类目前已实现GetEnumerator()使类本身可枚举
    public IEnumerator<string> GetEnumerator()//迭代器
    {
        yield return "black";
        yield return "gray";
        yield return "white";
    }
}
class Program
{
    static void Main()
    {
        var mc = new MyClass();
        foreach (string shade in mc)////该类目前已实现GetEnumerator()使类本身可枚 所以 写mc
        {
            Console.WriteLine(shade);
        }
    }
}

 

反编译IL代码

 

 

 

 

 

 

下图演示了MyClass的代码及产生的对象。注意编译器为我们自动做了多少工作。

  • 图左的迭代器代码演示了它的返回类型是IEnumerator<string>
  • 图右演示了它有一个嵌套类实现了IEnumerator<string>

 

 我们发现show类,内部自动生成一个枚举器,该枚举带了状态机的功能。并且自动继承了IEnumerator<object>, IEnumerator, IDisposable三个接口。

 IEnumerator<T> 和 IEnumerator 接口的类图

 

 

 

至此我们一目了然,迭代器就是带了状态机的枚举器。

 使用迭代器来创建可枚举类型

之前示例创建的类包含两部分:产生返回枚举器方法的迭代器以及返回枚举器的GetEnumerator方法。
本节例子中,我们用迭代器来创建可枚举类型,而不是枚举器。与之前的示例相比,本例有以下不同:

  • 若实现GetEnumerator,让它调用迭代器方法以获取自动生成的实现IEnumerable的类实例。然后从IEnumerable对象返回由GetEnumerator创建的枚举器,如图右
  • 若通过不实现GetEnumerator使类本身不可枚举,仍然可以使用由迭代器返回的可枚举类,只需要直接调用迭代器方法

本例中,BlackAndWhite迭代器方法返回IEnumerable<string>而不是IEnumerator<string>。因此MyClass首先调用BlackAndWhite方法获取它的可枚举类型对象,然后调用对象的GetEnumerator方法来获取结果,从而实现GetEnumerator方法

 

class MyClass
{
    //该类目前未实现 GetEnumerator()
    /*  public IEnumerator<string> GetEnumerator()
      {
          IEnumerable<string> myEnumerable = BlackAndWhite();
          return myEnumerable.GetEnumerator();
      }*
 public IEnumerable<string> BlackAndWhite()//迭代器  {
yield return "black"; yield return "gray"; yield return "white"; } } class Program { static void Main() { var mc = new MyClass(); //该类目前未实现GetEnumerator()使类本身不可枚举,仍然可以使用由迭代器返回的可枚举类,只需要直接调用迭代器方法 foreach (string shade in mc.BlackAndWhite()) { Console.Write(shade); } } }

 

反编译IL代码

 

 

迭代器实质


如下是需要了解的有关迭代器的其他重要事项。

  • 迭代器需要System.Collections.Generic命名空间
  • 在编译器生成的枚举器中,Reset方法没有实现。而它是接口需要的方法,因此调用时总是抛出System.NetSupportedException异常。

在后台,由编译器生成的枚举器类是包含4个状态的状态机。

  • Before 首次调用MoveNext的初始状态
  • Running 调用MoveNext后进入这个状态。在这个状态中,枚举器检测并设置下一项的为知。在遇到yield return、yield break或在迭代器体结束时,退出状态
  • Suspended 状态机等待下次调用MoveNext的状态
  • After 没有更多项可以枚举

如果状态机在Before或Suspended状态时调用MoveNext方法,就转到了Running状态。在Running状态中,它检测集合的下一项并设置为知。
如果有更多项,状态机会转入Suspended状态,如果没有更多项,它转入并保持在After状态。

 

练习题:用迭代器现实以下模型

 

源代码

 

posted @ 2022-01-30 17:12  小林野夫  阅读(1022)  评论(0编辑  收藏  举报
原文链接:https://www.cnblogs.com/cdaniu/