C# 枚举和迭代器

声明

本文内容来自微软 MVP solenovex 的视频教程——真会C#? - 第4章 进阶C#其它内容,大致和第 3 课—— 4.6 枚举和迭代器 对应。可在 GitHub 中查看 C# 视频教程的配套PPT

本文主要包括以下内容:

  1. 枚举器 Enumerator
  2. 集合初始化器
  3. 迭代器 Iterators
  4. 迭代器的语义
  5. 组合序列

枚举器 Enumerator

枚举器是一个只读的,作用于一序列值的、只能向前的游标。枚举器是一个实现了下列任意一个接口的对象:

System.Collections.IEnumerator
System.Collections.Generic.IEnumerator<T> 

技术上来说,任何一个含有名为MoveNext方法和名为Current的属性的对象,都会被当作枚举器来对待。foreach语句会迭代可枚举的对象(enumerable object)。可枚举的对象是一序列值的逻辑表示。它本身不是游标,它是一个可以基于本身产生游标的对象。

可枚举对象 enumerable object

一个可枚举对象可以是(下列任意一个):

  • 实现了IEnumerable或者IEnumerable<T>的对象
  • 有一个名为GetEnumerator的方法,并且该方法返回一个枚举器(enumerator)

IEnumeratorIEnumerable 是定义在 System.Collections 命名空间下的。IEnumerator<T>IEnumerable<T> 是定义在 System.Collections.Generic 命名空间下的。

枚举模式 enumeration pattern

class Enumerator // Typically implements IEnumerator or IEnumerator<T>
{
    public IteratorVariableType Current { get {...} }
    public bool MoveNext() {...}
}

class Enumerable // Typically implements IEnumerable or IEnumerable<T>
{
    public Enumerator GetEnumerator() {...}
}

注意:如果枚举器(enumerator)实现了IDisposable接口,那么foreach语句就会像
using语句那样,隐式的dispose掉这个 enumerator 对象。

foreach (char c in "beer")
    Console.WriteLine (c);
    
using (var enumerator = "beer".GetEnumerator())
    while (enumerator.MoveNext())
    {
        var element = enumerator.Current;
        Console.WriteLine (element);
    }

集合初始化器

你可以只用一步就把可枚举对象进行实例化并且填充里面的元素:

using System.Collections.Generic;
...
List<int> list = new List<int> {1, 2, 3};

但是编译器会把它翻译成:

using System.Collections.Generic;
...
List<int> list = new List<int>();
list.Add (1);
list.Add (2);
list.Add (3);

上例中,要求可枚举对象实现了System.Collections.IEnumerable接口,并且他还有一个可接受适当参数的Add方法。

var dict = new Dictionary<int, string>()
{
    { 5, "five" },
    { 10, "ten" }
};

// succinctly
var dict = new Dictionary<int, string>()
{
    [3] = "three",
    [10] = "ten"
};

迭代器 Iterators

foreach语句是枚举器(enumerator)的消费者,而迭代器(iterator)是枚举器的生产者。

yield return语句表达的意思是:这是你向我请求从枚举器产生的下一个元素。每逢遇到yield 语句,控制权都会回归到调用者那里,但是被调用这的状态还是会保持的,这样的话可以保证当调用者列举出下一个元素的时候,方法可以继续执行。这个状态的生命周期被绑定到了枚举器上。这样的话,当调用者完成枚举动作之后,状态可以被释放。

using System;
using System.Collections.Generic;
class Test
{
    static void Main()
    {
        foreach (int fib in Fibs(6))
            Console.Write (fib + " ");
    }
    
    static IEnumerable<int> Fibs (int fibCount)
    {
        for (int i = 0, prevFib = 1, curFib = 1; i < fibCount; i++)
        {
            yield return prevFib;
            int newFib = prevFib+curFib;
            prevFib = curFib;
            curFib = newFib;
        }
    }
}

// OUTPUT: 1 1 2 3 5 8

原理解释

编译器把迭代方法转换成私有的、实现了IEnumerable<T>IEnumerator<T>的类。迭代器块内部的逻辑被反转并且被切分到编译器生成的枚举器类里面的MoveNext方法和Current属性里。这意味着当你调用迭代器方法时,你所做的实际就是对编译器生成的类进行实例化;运行的代码里没有一行是你写的。你写的代码仅会在对结果序列进行枚举的时候才会运行,例如使用foreach语句。

迭代器的语义

迭代器(iterator)是含有一个或多个yield语句的方法、属性或索引器。迭代器必须返回下面四个接口中的一个(否则编译器会报错):

  • System.Collections.IEnumerable // Enumerable interfaces
  • System.Collections.Generic.IEnumerable<T> // Enumerable interfaces
  • System.Collections.IEnumerator // Enumerator interfaces
  • System.Collections.Generic.IEnumerator<T> // Enumerator interfaces

根据迭代器返回的是enumerable接口还是enumerator接口,迭代器会拥有不同的语义。

多个yield语句

方法里可以含有多个yield语句:

class Test
{
    static void Main()
    {
        foreach (string s in Foo())
            Console.WriteLine(s); // Prints "One","Two","Three"
    }
    
    static IEnumerable<string> Foo()
    {
        yield return "One";
        yield return "Two";
        yield return "Three";
    }
}

yield break

yield break语句表示迭代器块会提前退出,不再返回更多的元素。

static IEnumerable<string> Foo (bool breakEarly)
{
    yield return "One";
    yield return "Two";
    if (breakEarly)
        yield break;
    yield return "Three";
}

return语句在迭代器块里面是非法的,你必须使用yield break代替。

迭代器和try/catch/finally

yield return语句不可以出现在含有catch子句的try块里面:

IEnumerable<string> Foo()
{
    try { yield return "One"; } // Illegal
    catch { ... }
}

yield return 也不能出现在catch或者finally块里面。但是 yield return可以出现在只含有finally块的try块里面:

IEnumerable<string> Foo()
{
    try { yield return "One"; } // OK
    finally { ... }
}

当消费的枚举器到达序列终点或被disposed的时候,finally块里面的代码会执行。如果你提前进行了break,那么foreach语句也会dispose掉枚举器,所以用起来很安全。

当显式的使用枚举器的时候,通常会犯这样一个错误:没有dispose掉枚举就不再用它了,这就绕开了finally块。针对这种情况,你可以使用using语句来规避风险:

string firstElement = null;
var sequence = Foo();
using (var enumerator = sequence.GetEnumerator())
    if (enumerator.MoveNext())
        firstElement = enumerator.Current;

组合序列

迭代器是高度可组合的。

using System;
using System.Collections.Generic;
class Test
{
    static void Main()
    {
        foreach (int fib in EvenNumbersOnly (Fibs(6)))
            Console.WriteLine (fib);
    }
    
    static IEnumerable<int> Fibs (int fibCount)
    {
        for (int i = 0, prevFib = 1, curFib = 1; i < fibCount; i++)
        {
            yield return prevFib;
            int newFib = prevFib+curFib;
            prevFib = curFib;
            curFib = newFib;
        }
    }
    
    static IEnumerable<int> EvenNumbersOnly (IEnumerable<int> sequence)
    {
        foreach (int x in sequence)
            if ((x % 2) == 0)
                yield return x;
    }
}

上例中的每个元素直到最后时刻才会被计算,也就是被MoveNext()操作请求的时候。

Composing sequences

参考

Iterators (C#)
System.Collections.Generic Namespace
IEnumerable Interface
IEnumerator Interface
yield (C# Reference)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容

  • 一、数组数组是一组使用数字索引的对象,这些对象属于同一种类型。虽然C#为创建数组提供了直接的语言支持,但通用类型系...
    CarlDonitz阅读 667评论 0 1
  • 1.迭代器   首先我们要谈的就是迭代器,很多情况下我们都使用了迭代器,并不仅仅是因为协程,当我们使用foreac...
    joshuaAS阅读 2,846评论 1 4
  • 在这个降低入门门槛的大环境下,Unity 因为考虑到降低门槛,设计之初就是一个单线程,不允许在另外的线程中进行渲染...
    耳朵里有只风阅读 8,267评论 0 5
  • 集合是.NET FCL(Framework Class Library)的重要组成部分,我们平常撸C#代码时免不了...
    aslbutton阅读 945评论 0 50
  • 数据结构 一般将数据结构分为两大类: 线性数据结构和非线性数据结构。 线性数据结构有: 线性表、栈、队列、串、数组...
    沉麟阅读 890评论 0 0