C# 枚舉和迭代器

聲明

本文內(nèi)容來自微軟 MVP solenovex 的視頻教程——真會(huì)C#? - 第4章 進(jìn)階C#其它內(nèi)容过椎,大致和第 3 課—— 4.6 枚舉和迭代器 對(duì)應(yīng)室梅。可在 GitHub 中查看 C# 視頻教程的配套PPT

本文主要包括以下內(nèi)容:

  1. 枚舉器 Enumerator
  2. 集合初始化器
  3. 迭代器 Iterators
  4. 迭代器的語義
  5. 組合序列

枚舉器 Enumerator

枚舉器是一個(gè)只讀的潭流,作用于一序列值的竞惋、只能向前的游標(biāo)。枚舉器是一個(gè)實(shí)現(xiàn)了下列任意一個(gè)接口的對(duì)象:

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

技術(shù)上來說灰嫉,任何一個(gè)含有名為MoveNext方法和名為Current的屬性的對(duì)象,都會(huì)被當(dāng)作枚舉器來對(duì)待嗓奢。foreach語句會(huì)迭代可枚舉的對(duì)象(enumerable object)讼撒。可枚舉的對(duì)象是一序列值的邏輯表示股耽。它本身不是游標(biāo)根盒,它是一個(gè)可以基于本身產(chǎn)生游標(biāo)的對(duì)象。

可枚舉對(duì)象 enumerable object

一個(gè)可枚舉對(duì)象可以是(下列任意一個(gè)):

  • 實(shí)現(xiàn)了IEnumerable或者IEnumerable<T>的對(duì)象
  • 有一個(gè)名為GetEnumerator的方法物蝙,并且該方法返回一個(gè)枚舉器(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)實(shí)現(xiàn)了IDisposable接口诬乞,那么foreach語句就會(huì)像
using語句那樣册赛,隱式的dispose掉這個(gè) enumerator 對(duì)象钠导。

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

集合初始化器

你可以只用一步就把可枚舉對(duì)象進(jìn)行實(shí)例化并且填充里面的元素:

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

但是編譯器會(huì)把它翻譯成:

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

上例中,要求可枚舉對(duì)象實(shí)現(xiàn)了System.Collections.IEnumerable接口森瘪,并且他還有一個(gè)可接受適當(dāng)參數(shù)的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)的消費(fèi)者,而迭代器(iterator)是枚舉器的生產(chǎn)者扼睬。

yield return語句表達(dá)的意思是:這是你向我請(qǐng)求從枚舉器產(chǎn)生的下一個(gè)元素逮栅。每逢遇到yield 語句,控制權(quán)都會(huì)回歸到調(diào)用者那里窗宇,但是被調(diào)用這的狀態(tài)還是會(huì)保持的措伐,這樣的話可以保證當(dāng)調(diào)用者列舉出下一個(gè)元素的時(shí)候,方法可以繼續(xù)執(zhí)行军俊。這個(gè)狀態(tài)的生命周期被綁定到了枚舉器上侥加。這樣的話,當(dāng)調(diào)用者完成枚舉動(dòng)作之后蝇完,狀態(tài)可以被釋放官硝。

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

原理解釋

編譯器把迭代方法轉(zhuǎn)換成私有的、實(shí)現(xiàn)了IEnumerable<T>IEnumerator<T>的類短蜕。迭代器塊內(nèi)部的邏輯被反轉(zhuǎn)并且被切分到編譯器生成的枚舉器類里面的MoveNext方法和Current屬性里氢架。這意味著當(dāng)你調(diào)用迭代器方法時(shí),你所做的實(shí)際就是對(duì)編譯器生成的類進(jìn)行實(shí)例化朋魔;運(yùn)行的代碼里沒有一行是你寫的岖研。你寫的代碼僅會(huì)在對(duì)結(jié)果序列進(jìn)行枚舉的時(shí)候才會(huì)運(yùn)行,例如使用foreach語句警检。

迭代器的語義

迭代器(iterator)是含有一個(gè)或多個(gè)yield語句的方法孙援、屬性或索引器。迭代器必須返回下面四個(gè)接口中的一個(gè)(否則編譯器會(huì)報(bào)錯(cuò)):

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

根據(jù)迭代器返回的是enumerable接口還是enumerator接口扇雕,迭代器會(huì)擁有不同的語義拓售。

多個(gè)yield語句

方法里可以含有多個(gè)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語句表示迭代器塊會(huì)提前退出,不再返回更多的元素镶奉。

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語句不可以出現(xiàn)在含有catch子句的try塊里面:

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

yield return 也不能出現(xiàn)在catch或者finally塊里面哨苛。但是 yield return可以出現(xiàn)在只含有finally塊的try塊里面:

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

當(dāng)消費(fèi)的枚舉器到達(dá)序列終點(diǎn)或被disposed的時(shí)候鸽凶,finally塊里面的代碼會(huì)執(zhí)行。如果你提前進(jìn)行了break建峭,那么foreach語句也會(huì)dispose掉枚舉器玻侥,所以用起來很安全。

當(dāng)顯式的使用枚舉器的時(shí)候亿蒸,通常會(huì)犯這樣一個(gè)錯(cuò)誤:沒有dispose掉枚舉就不再用它了凑兰,這就繞開了finally塊掌桩。針對(duì)這種情況,你可以使用using語句來規(guī)避風(fēng)險(xiǎn):

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;
    }
}

上例中的每個(gè)元素直到最后時(shí)刻才會(huì)被計(jì)算拘鞋,也就是被MoveNext()操作請(qǐng)求的時(shí)候。

Composing sequences

參考

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

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末矢门,一起剝皮案震驚了整個(gè)濱河市盆色,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌祟剔,老刑警劉巖隔躲,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異物延,居然都是意外死亡宣旱,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門叛薯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來浑吟,“玉大人,你說我怎么就攤上這事耗溜∽榱Γ” “怎么了?”我有些...
    開封第一講書人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵抖拴,是天一觀的道長(zhǎng)燎字。 經(jīng)常有香客問我,道長(zhǎng)阿宅,這世上最難降的妖魔是什么候衍? 我笑而不...
    開封第一講書人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮洒放,結(jié)果婚禮上蛉鹿,老公的妹妹穿的比我還像新娘。我一直安慰自己往湿,他們只是感情好榨为,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著煌茴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪日川。 梳的紋絲不亂的頭發(fā)上蔓腐,一...
    開封第一講書人閱讀 51,301評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音龄句,去河邊找鬼回论。 笑死散罕,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的傀蓉。 我是一名探鬼主播欧漱,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼葬燎!你這毒婦竟也來了误甚?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤谱净,失蹤者是張志新(化名)和其女友劉穎窑邦,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體壕探,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡冈钦,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了李请。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瞧筛。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖导盅,靈堂內(nèi)的尸體忽然破棺而出较幌,到底是詐尸還是另有隱情,我是刑警寧澤认轨,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布绅络,位于F島的核電站,受9級(jí)特大地震影響嘁字,放射性物質(zhì)發(fā)生泄漏恩急。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一纪蜒、第九天 我趴在偏房一處隱蔽的房頂上張望衷恭。 院中可真熱鬧,春花似錦纯续、人聲如沸随珠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)窗看。三九已至,卻和暖如春倦炒,著一層夾襖步出監(jiān)牢的瞬間显沈,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留拉讯,地道東北人涤浇。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像魔慷,于是被迫代替她去往敵國(guó)和親只锭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 一院尔、數(shù)組數(shù)組是一組使用數(shù)字索引的對(duì)象蜻展,這些對(duì)象屬于同一種類型。雖然C#為創(chuàng)建數(shù)組提供了直接的語言支持召边,但通用類型系...
    CarlDonitz閱讀 667評(píng)論 0 1
  • 1.迭代器 ??首先我們要談的就是迭代器铺呵,很多情況下我們都使用了迭代器,并不僅僅是因?yàn)閰f(xié)程隧熙,當(dāng)我們使用foreac...
    joshuaAS閱讀 2,848評(píng)論 1 4
  • 在這個(gè)降低入門門檻的大環(huán)境下,Unity 因?yàn)榭紤]到降低門檻躏敢,設(shè)計(jì)之初就是一個(gè)單線程闷愤,不允許在另外的線程中進(jìn)行渲染...
    耳朵里有只風(fēng)閱讀 8,269評(píng)論 0 5
  • 集合是.NET FCL(Framework Class Library)的重要組成部分,我們平常擼C#代碼時(shí)免不了...
    aslbutton閱讀 945評(píng)論 0 50
  • 數(shù)據(jù)結(jié)構(gòu) 一般將數(shù)據(jù)結(jié)構(gòu)分為兩大類: 線性數(shù)據(jù)結(jié)構(gòu)和非線性數(shù)據(jù)結(jié)構(gòu)件余。 線性數(shù)據(jù)結(jié)構(gòu)有: 線性表讥脐、棧、隊(duì)列啼器、串旬渠、數(shù)組...
    沉麟閱讀 892評(píng)論 0 0