解析c# yield關(guān)鍵字
1.yield實(shí)現(xiàn)的功能
yield return:
先看下面的代碼,通過yield return實(shí)現(xiàn)了類似用foreach遍歷數(shù)組的功能,說明yield return也是用來實(shí)現(xiàn)迭代器的功能的。
using static System.Console;
using System.Collections.Generic;
class Program
{
//一個(gè)返回類型為IEnumerable<int>,其中包含三個(gè)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();
}
}
輸出結(jié)果:
1
2
3
yield break:
再看下面的代碼,只輸出了1,2,沒有輸出3,說明這個(gè)迭代器被yield break停掉了,所以yield break是用來終止迭代的。
using static System.Console;
using System.Collections.Generic;
class Program
{
//一個(gè)返回類型為IEnumerable<int>,其中包含三個(gè)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();
}
}
輸出結(jié)果:
1
2
2.只能使用在返回類型必須為 IEnumerable、IEnumerable<T>、IEnumerator 或 IEnumerator<T>的方法、運(yùn)算符、get訪問器中。
3.yield關(guān)鍵字的實(shí)現(xiàn)原理
我們用while循環(huán)代替foreach循環(huán),發(fā)現(xiàn)我們雖然沒有實(shí)現(xiàn)GetEnumerator(),也沒有實(shí)現(xiàn)對(duì)應(yīng)的IEnumerator的MoveNext(),和Current屬性,但是我們?nèi)匀荒苷J褂眠@些函數(shù)。
class Program
{
//一個(gè)返回類型為IEnumerable<int>,其中包含三個(gè)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();
}
}
輸出結(jié)果:
1
2
3
至于為什么會(huì)出現(xiàn)這種情況,我們可以用ILSpy對(duì)生成的exe進(jìn)行反編譯來找到原因。
由于直接反編譯成C#會(huì)變?yōu)樵瓨?/p>

所以我們選擇反編譯為帶C#注釋的IL代碼,雖然可讀性差點(diǎn),但是可以詳細(xì)的了解其中過的原理。
先來看Program翻譯的情況,編譯的時(shí)候自動(dòng)生成了一個(gè)新的類。

接下來我們來仔細(xì)看這些代碼,EnumerableFuc()返回了這個(gè)新的類。

看這個(gè)代碼自動(dòng)生成的類的實(shí)現(xiàn),發(fā)現(xiàn)它繼承了IEnumerable、IEnumerable<T>、IEnumerator 或 IEnumerator<T>,這時(shí)我們應(yīng)該已經(jīng)能猜到這個(gè)新的類就是我們沒有實(shí)現(xiàn)對(duì)應(yīng)的IEnumerator的MoveNext(),和Current屬性,但是我們?nèi)匀荒苷J褂眠@些函數(shù)的原因了。

我們?cè)賮砜匆幌逻@個(gè)類具體是如何實(shí)現(xiàn)迭代的呢,我們主要來看一下MoveNext()函數(shù)


每次調(diào)用MoveNext()函數(shù)都會(huì)將state加1,一共進(jìn)行了4次迭代,前三次返回true,最后一次返回false,代表迭代結(jié)束。這四次迭代對(duì)應(yīng)被3個(gè)yield return語句分成4部分的enumberableFuc()中的語句。
用enumberableFuc()來進(jìn)行迭代的真實(shí)流程就是:
1.運(yùn)行enumberableFuc()函數(shù),獲取代碼自動(dòng)生成的類的實(shí)例。
2.接著調(diào)用GetEnumberator()函數(shù),將獲取的類自己作為迭代器開始迭代。
3.每次運(yùn)行MoveNext(),state增加1,通過switch語句可以讓每次調(diào)用MoveNext()的時(shí)候執(zhí)行不同部分的代碼。
4。MoveNext()返回false,結(jié)束。
這也能說明yield關(guān)鍵字其實(shí)是一種語法糖,最終還是通過實(shí)現(xiàn)IEnumberable<T>、IEnumberable、IEnumberator<T>和IEnumberator接口實(shí)現(xiàn)的迭代功能。
以上就是解析c# yield關(guān)鍵字的詳細(xì)內(nèi)容,更多關(guān)于c# yield關(guān)鍵字的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳細(xì)聊聊C#的并發(fā)機(jī)制優(yōu)秀在哪
并發(fā)其實(shí)是一個(gè)很泛的概念,字面意思就是"同時(shí)做多件事",不過方式有所不同,下面這篇文章主要給大家介紹了關(guān)于C#并發(fā)機(jī)制的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-02-02
C#實(shí)現(xiàn)啟動(dòng)項(xiàng)管理的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用C#實(shí)現(xiàn)啟動(dòng)項(xiàng)管理,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C#有一定的幫助,感興趣的小伙伴可以了解一下2022-12-12
C# MJPEG 客戶端簡(jiǎn)單實(shí)現(xiàn)方法
這篇文章主要介紹了C# MJPEG 客戶端簡(jiǎn)單實(shí)現(xiàn)的方法,幫助大家更好的理解和學(xué)習(xí)使用c#,感興趣的朋友可以了解下2021-03-03
Unity Sockect實(shí)現(xiàn)畫面實(shí)時(shí)傳輸案例原理解析
Socket是比較常用的一種通信方式,本文通過案例給大家介紹Unity Sockect實(shí)現(xiàn)畫面實(shí)時(shí)傳輸功能,感興趣的朋友一起看看吧2021-08-08
.net中前臺(tái)javascript與后臺(tái)c#函數(shù)相互調(diào)用問題
.net中前臺(tái)javascript與后臺(tái)c#函數(shù)相互調(diào)用問題...2007-12-12

