解析c# yield關鍵字
1.yield實現(xiàn)的功能
yield return:
先看下面的代碼,通過yield return實現(xiàn)了類似用foreach遍歷數(shù)組的功能,說明yield return也是用來實現(xiàn)迭代器的功能的。
using static System.Console; using System.Collections.Generic; class Program { //一個返回類型為IEnumerable<int>,其中包含三個yield return public static IEnumerable<int> enumerableFuc() { yield return 1; yield return 2; yield return 3; } static void Main(string[] args) { //通過foreach循環(huán)迭代此函數(shù) foreach(int item in enumerableFuc()) { WriteLine(item); } ReadKey(); } }
輸出結果:
1
2
3
yield break:
再看下面的代碼,只輸出了1,2,沒有輸出3,說明這個迭代器被yield break停掉了,所以yield break是用來終止迭代的。
using static System.Console; using System.Collections.Generic; class Program { //一個返回類型為IEnumerable<int>,其中包含三個yield return public static IEnumerable<int> enumerableFuc() { yield return 1; yield return 2; yield break; yield return 3; } static void Main(string[] args) { //通過foreach循環(huán)迭代此函數(shù) foreach(int item in enumerableFuc()) { WriteLine(item); } ReadKey(); } }
輸出結果:
1
2
2.只能使用在返回類型必須為 IEnumerable、IEnumerable<T>、IEnumerator 或 IEnumerator<T>的方法、運算符、get訪問器中。
3.yield關鍵字的實現(xiàn)原理
我們用while循環(huán)代替foreach循環(huán),發(fā)現(xiàn)我們雖然沒有實現(xiàn)GetEnumerator(),也沒有實現(xiàn)對應的IEnumerator的MoveNext(),和Current屬性,但是我們?nèi)匀荒苷J褂眠@些函數(shù)。
class Program { //一個返回類型為IEnumerable<int>,其中包含三個yield return public static IEnumerable<int> enumerableFuc() { yield return 1; yield return 2; yield return 3; } static void Main(string[] args) { //用while循環(huán)代替foreach IEnumerator<int> enumerator = enumerableFuc().GetEnumerator(); while (enumerator.MoveNext()) { int current = enumerator.Current; WriteLine(current); } ReadKey(); } }
輸出結果:
1
2
3
至于為什么會出現(xiàn)這種情況,我們可以用ILSpy對生成的exe進行反編譯來找到原因。
由于直接反編譯成C#會變?yōu)樵瓨?/p>
所以我們選擇反編譯為帶C#注釋的IL代碼,雖然可讀性差點,但是可以詳細的了解其中過的原理。
先來看Program翻譯的情況,編譯的時候自動生成了一個新的類。
接下來我們來仔細看這些代碼,EnumerableFuc()返回了這個新的類。
看這個代碼自動生成的類的實現(xiàn),發(fā)現(xiàn)它繼承了IEnumerable、IEnumerable<T>、IEnumerator 或 IEnumerator<T>,這時我們應該已經(jīng)能猜到這個新的類就是我們沒有實現(xiàn)對應的IEnumerator的MoveNext(),和Current屬性,但是我們?nèi)匀荒苷J褂眠@些函數(shù)的原因了。
我們再來看一下這個類具體是如何實現(xiàn)迭代的呢,我們主要來看一下MoveNext()函數(shù)
每次調(diào)用MoveNext()函數(shù)都會將state加1,一共進行了4次迭代,前三次返回true,最后一次返回false,代表迭代結束。這四次迭代對應被3個yield return語句分成4部分的enumberableFuc()中的語句。
用enumberableFuc()來進行迭代的真實流程就是:
1.運行enumberableFuc()函數(shù),獲取代碼自動生成的類的實例。
2.接著調(diào)用GetEnumberator()函數(shù),將獲取的類自己作為迭代器開始迭代。
3.每次運行MoveNext(),state增加1,通過switch語句可以讓每次調(diào)用MoveNext()的時候執(zhí)行不同部分的代碼。
4。MoveNext()返回false,結束。
這也能說明yield關鍵字其實是一種語法糖,最終還是通過實現(xiàn)IEnumberable<T>、IEnumberable、IEnumberator<T>和IEnumberator接口實現(xiàn)的迭代功能。
以上就是解析c# yield關鍵字的詳細內(nèi)容,更多關于c# yield關鍵字的資料請關注腳本之家其它相關文章!
相關文章
C# 9 中新加入的關鍵詞 init,record,with
這篇文章主要介紹了C# 9 中新加入的關鍵詞 init,record,with的相關資料,幫助大家更好的理解和學習c# 9,感興趣的朋友可以了解下2020-08-08C# XML基礎入門小結(XML文件內(nèi)容增刪改查清)
本文主要介紹了C# XML基礎入門小結(XML文件內(nèi)容增刪改查清),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-04-04C# WebApi Get請求方式傳遞實體參數(shù)的方法示例
這篇文章主要給大家介紹了關于C# WebApi Get請求方式傳遞實體參數(shù)的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用C#具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2019-04-04C#利用Label標簽控件模擬窗體標題的移動及窗體顏色不斷變換效果
Label標簽控件相信對大家來說都不陌生,下面這篇文章主要給大家介紹了關于C#利用Label標簽控件模擬窗體標題的移動及窗體顏色不斷變換效果的相關資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下。2017-12-12