C#中yield的使用

來源:互聯網
上載者:User

標籤:工作   匿名方法   ros   div   pip   pes   有趣   tor   它的   

yield 關鍵字向編譯器指示它所在的方法是迭代器塊。編譯器產生一個類來實現迭代器塊中表示的行為。在迭代器塊中,yield 關鍵字與 return 關鍵字結合使用,向列舉程式對象提供值。這是一個傳回值,例如,在 foreach 語句的每一次迴圈中返回的值。yield 關鍵字也可與 break 結合使用,表示迭代結束。

例子:
yield return <expression>;
yield break;

在 yield return 語句中,將計算 expression 並將結果以值的形式返回給列舉程式對象;expression 必須可以隱式轉換為 yield 類型的迭代器。

在 yield break 語句中,控制權將無條件地返回給迭代器的調用方,該調用方為列舉程式對象的 IEnumerator.MoveNext 方法(或其對應的泛型System.Collections.Generic.IEnumerable<T>)或 Dispose 方法。

yield 語句只能出現在 iterator 塊中,這種塊可作為方法、運算子或訪問器的主體實現。這類方法、運算子或訪問器的體受以下約束的控制:

  • 不允許不安全塊。

  • 方法、運算子或訪問器的參數不能是 ref 或 out。

  • yield return 語句不能放在 try-catch 塊中的任何位置。該語句可放在後跟 finally 塊的 try 塊中。

  • yield break 語句可放在 try 塊或 catch 塊中,但不能放在 finally 塊中。

yield 語句不能出現在匿名方法中。有關更多資訊,請參見匿名方法(C# 編程指南)。

當和 expression 一起使用時,yield return 語句不能出現在 catch 塊中或含有一個或多個 catch 子句的 try 塊中。

 

對於”yield”這個關鍵字我已經見過N次了,直到最近我才知道這個關鍵字所蘊含的力量。我將在下面展示出一些使用”yield”讓你的代碼有更高可讀性和更好效能的例子.

 

為了讓你對yield有一些快速概覽,我首先要展示一個沒有使用這個關鍵字的例子,下面的代碼很簡單,但在我最近的項目中卻很常見

IList<string> FindBobs(IEnumerable<string> names){var bobs = new List<string>();foreach(var currName in names){if(currName == "Bob")bobs.Add(currName);}return bobs;}

注意在這裡我使用IEnumerable<string>作為參數類型並以IList<string>作為傳回型別,通常來說,我更傾向於在參數輸入的類型方面的範圍越寬越好,但在傳回型別上面更加嚴格(譯者按:即輸入時多用基類或介面,返回時用子類或實作類別),對於輸入來說,如果你需要用foreach來對其進行迴圈的話,使用IEnumerable會更有意義。而對於輸出(譯者按:也就是返回),我使用介面來讓實現部分可以改變。在這裡我想讓調用者省去產生列表的麻煩,所以我選擇list作為傳回型別.

 

而問題在於,我的設計並不具有可連結性,這樣的設計需要產生列表作為傳回值,實現上,這個列表或許不會很大,但這並不必要

 

現在,讓我們來看看以“yield”的方式來做這些,而後我會解釋如何使用它,以及它工作的原理。

IEnumerable<string> FindBobs(IEnumerable<string> names){foreach(var currName in names){if(currName == "Bob")yield return currName;}}

在這個版本中,我們將傳回型別改為IEnumerable,並且我們使用”yield return”.注意我再也不需要建立一個列表,現在是不是有些迷惑的?別著急,在理解它的工作方式的情況它會變的越來越簡單.

當使用”yield return”關鍵片語時,.net會為你產生一大串管道代碼,你可以儘管假裝這是個魔法。當開始在被調用的代碼中迴圈時(這裡不是list),實現上發生的是這個函數被一遍一遍的調用,但每一次都從上一次執行退出的部分開始繼續執行.

 

傳統的執行方法
  1. 調用函數
  2. 函數執行並返回list
  3. 調用部分使用返回的list
Yield的執行方法
  1. 調用函數
  2. 調用者請求item
  3. 下一個item返回
  4. 回到步驟2

雖然yield執行的實現貌似有些複雜,但我們最終只需要一次“彈出”一個item,而不是建立整個list並返回.

對於句法說,我個人認為yield更加簡潔,並且對於傳遞函數的用途表現的更好(譯者按:也就是代碼可讀性),我使用IEnumerable作為傳回型別來通知調用者它可以被foreach迴圈並且返回資料,而調用者現在可以自己決定它是否願意將傳回值存放到列表中,即使這會以效能作為代價。

在我提供的這個簡單例子中,也許你並不能發現很多使用yield的好處,然而,你可以在調用者需要取消遍曆所有的函數提供的內容時節省很多不必要的工作,當你在方法連結時使用yield,你可以省下的工作(時間)或許會成倍疊加。

Ayende已經有了很棒的例子:using yield for a slick pipes & filters implementation,他甚至還講述了:version that is multi-threaded。這讓我覺得非常有趣.

最開始我對yield的保留意見是使用這個關鍵字或許會導致潛在的效能問題,但實際上,至今為止我還未能發現任何資訊來說明關於yield對效能的影響,而我在上面提到提高效能的地方遠遠大於編譯器overhead那部分。

C#中yield的使用

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.