c yield關鍵字原理詳解

2021-08-24 20:15:28 字數 2364 閱讀 1600

1.yield實現的功能

yield return:

先看下面的**,通過yield return實現了類似用foreach遍歷陣列的功能,說明yield return也是用來實現迭代器的功能的。

using static system.console;

using system.collections.generic;

class program

static void main(string args)

readkey();

}}輸出結果:12

3

yield break:再看下面的**,只輸出了1,2,沒有輸出3,說明這個迭代器被yield break停掉了,所以yield break是用來終止迭代的。

using static system.console;

using system.collections.generic;

class program

static void main(string args)

readkey();

}}輸出結果:

12

2.只能使用在返回型別必須為 ienumerable、ienumerable、ienumerator 或 ienumerator的方法、運算子、get訪問器中。3.yield關鍵字的實現原理我們用while迴圈代替foreach迴圈,發現我們雖然沒有實現getenumerator(),也沒有實現對應的ienumerator的movenext(),和current屬性,但是我們仍然能正常使用這些函式。

class program

static void main(string args)

readkey();

}}輸出結果:12

3

至於為什麼會出現這種情況,我們可以用ilspy對生成的exe進行反編譯來找到原因。

由於直接反編譯成c#會變為原樣

所以我們選擇反編譯為帶c#注釋的il**,雖然可讀性差點,但是可以詳細的了解其中過的原理。

先來看program翻譯的情況,編譯的時候自動生成了乙個新的類。

接下來我們來仔細看這些**,enumerablefuc()返回了這個新的類。

看這個**自動生成的類的實現,發現它繼承了ienumerable、ienumerable、ienumerator 或 ienumerator,這時我們應該已經能猜到這個新的類就是我們沒有實現對應的ienumerator的movenext(),和current屬性,但是我們仍然能正常使用這些函式的原因了。

我們再來看一下這個類具體是如何實現迭代的呢,我們主要來看一下movenext()函式

每次呼叫movenext()函式都會將state加1,一共進行了4次迭代,前三次返回true,最後一次返回false,代表迭代結束。這四次迭代對應被3個yield return語句分成4部分的enumberablefuc()中的語句。

用enumberablefuc()來進行迭代的真實流程就是:

1.執行enumberablefuc()函式,獲取**自動生成的類的例項。

2.接著呼叫getenumberator()函式,將獲取的類自己作為迭代器開始迭代。

3.每次執行movenext(),state增加1,通過switch語句可以讓每次呼叫movenext()的時候執行不同部分的**。

4。movenext()返回false,結束。

這也能說明yield關鍵字其實是一種語法糖,最終還是通過實現ienumberable、ienumberable、ienumberator和ienumberator介面實現的迭代功能。

c yield關鍵字原理詳解

1.yield實現的功能 yield return 先看下面的 通過yield return實現了類似用foreach遍歷陣列的功能,說明yield return也是用來實現迭代器的功能的。using static system.console using system.collections.ge...

C yield 關鍵字 使用

在語句中使用 yield 關鍵字,則指示在的方案 運算子或 get 訪問器是迭代器。使用的迭代器對集合的自定義迭代。使用乙個 yield return 語句返回每個元素乙個節點。例子 public class powersof2 i public static system.collections....

解析c yield關鍵字

1.yield實現的功能 yield return 先看下面的 通過yield return實現了類似用foreach遍歷陣列的功能,說明yield return也是用來實現迭代器的功能的。using static system.console using system.collections.ge...