22迭代器
22.1迭代器區塊
迭代器區塊就是產生值的有序序列的語句區塊。迭代器區塊透過一個或多個yield語句區別於常規語句區塊。
l yield return 語句產生迭代的下一個值。
l yield break 語句指明迭代完成。
迭代器區塊可以被用作一個方法體(method-body)、運算子體(operator-body)、存取器體(accessor-body),前提是對應函數成員的回傳類型是枚舉器(enumerator)介面之一或可枚舉(enumerable)介面之一。
迭代器塊在C#語法中不什麼獨特的元素。它們在幾個方面受到限制,並且主要的作用在函數成員聲明的語義上,但它們在語法上只是語句塊而已。
當一個函數成員使用迭代器區塊實作時,對於正式參數清單指定任何ref或out參數將導致編譯時錯誤。
return語句出現在迭代器區塊中會導致編譯時錯誤(但yield return語句是允許的)。
在迭代器區塊中包含不安全上下文(§18.1)將導致編譯時錯誤。即使是當迭代器宣告內嵌在不安全上下文中,迭代器區塊也總是定義為一個安全上下文。
22.1.1枚舉器介面
枚舉器介面(enumerator interface)[/b]是System.Collections.IEnumerator介面以及System.Collections.Generic.IEnumerator
22.1.2可枚舉介面
可枚舉介面(enumerable interface[/b])[/b]是System.Collections.IEnumerable介面和System.Collections.Generic.IEnumerable
22.1.3Yield 類型
迭代器區塊產生具有相同類型的所有值的序列。給型別被稱為迭代器區塊的yield[/b]型別(yield type[/b])[/b]。
l 迭代器區塊的yield類型通常用於實作傳回IEnumerator或IEnumerable是物件的函數成員。
l 迭代器區塊的yield類型通常用於實作傳回IEnumerator
22.1.4 this 存取
在類別的實例成員的迭代器區塊內,this表達式被歸類為值。該值的類型就是類別類型,在這個類型可以採用這種用法,這個值就是成員被呼叫時的物件的參考。
在結構的實例成員的迭代器區塊內,this表達時被歸類為一個變數。該變數的類型就是結構類型,在這個結構中它可以採用這種用法。此變數表示一個成員被呼叫時的對應結構的一個拷貝。在結構實例成員的迭代器區塊內,this變數的行為就好像是結構類型的一個值參數。
22.2枚舉物件
當傳回枚舉器介面類型的函數成員使用迭代器區塊實作時,呼叫函數成員並不會立即執行迭代器區塊中的程式碼。相反,枚舉器物件(enumerator object)將被建立和傳回。此物件封裝了在迭代器區塊中指定的程式碼,當枚舉器物件的MoveNext方法被呼叫時,迭代器區塊中的程式碼就會執行。枚舉器物件有如下的特徵。
l 它實作了IEnumerator和IEnumerator
l 它實作了System.IDisposable。
l 它被使用實參值(如果有的話)的拷貝而初始化,而實例值將被傳遞給函數成員。
l 它有四個潛在的狀態before、running、suspended和after,並且它在before狀態之前被初始化。
枚舉器物件通常是一個編譯器產生的枚舉器類別實例,它封裝了迭代器語句區塊中的程式碼,並且實作了枚舉器接口,但其它實作方法也是可以的。如果一個枚舉器類別是由編譯器產生的,這個類別將會是內嵌的,在包含函數成員的類別中,類別將具有私有可訪問性,並且該類別具有一個保留為編譯器所用的名字(§2.4.2)。
枚舉器物件可以實現比在此指定的更多介面。
隨後幾節描述了由IEnumerable和IEnumerable
請注意,枚舉器物件並不支援IEnumerator.Reset方法。呼叫該方法將會拋出System.NotSupportedException異常。
22.2.1MoveNext方法
枚舉器物件的MoveNext方法封裝了迭代器區塊的程式碼。呼叫MoveNext方法將執行迭代器內的程式碼,並將枚舉物件的Current屬性設為適當的值。由MoveNext方法執行的精確動作,取決於當MoveNext方法被呼叫時枚舉器物件的狀態。
l 如果枚舉器物件狀態是before,呼叫MoveNext
n 將把狀態改為running。
n 將把迭代器區塊的參數(包括this)初始化為,當枚舉器物件被初始化而保存的實參值和實例值。
n 從開始執行迭代器區塊直到執行中斷(如下面所描述的)。
l 如果枚舉器物件的狀態是running,呼叫MoveNext的結果是未指定的。
l 如果枚舉器物件的狀態是suspended,呼叫MoveNext
n 將把狀態改為running。
l 恢復所有局部變數和參數(包括this)的值為迭代器最後一次掛起(suspended)時執行狀態的值。請注意,由這些變數所引用的任何物件的內容,都可能因為前一次對MoveNext的呼叫而改變。
n 在引發執行暫停的yield return 語句之後重新開始執行迭代器區塊,而這個狀態會繼續直到執行被中斷(如下所描述)。
l 如果枚舉器物件的狀態是after,那麼呼叫MoveNext將會傳回false。
當MoveNext執行迭代器區塊時,有四種方法可以中斷執行:透過一個yield return 語句,透過一個yield break語句,到達迭代器區塊的結束點,以及一個異常被拋出,並被傳播到迭代器塊之外。
l 當遇到一個yield return 語句時(§22.4),將會發生如下情況
n 在該語句中被給定的表達式將被計算,隱式地轉換到產生類型(yield type),並被賦值給枚舉物件的Current屬性。
n 迭代器體的執行將被掛起。所有局部變數的值和參數(包括this)被保存,該yield return 語句的位置也被保存。如果yield return 語句在一個或多個try區塊之內,與之關聯的finally區塊在此時將不會執行。
n 枚舉器物件的狀態改為suspended。
n MoveNext方法對呼叫方傳回true,表示迭代器成功前進到下一個值。
l 當遇到yield break 語句時,將會發生如下情況
n 如果yield break 語句在一個或多個try區塊之內,與之關聯的finally語句將被執行。
n 枚舉器物件的狀態改為after。
n MoveNext方法對呼叫方傳回false,表示迭代已經完成。
l 當遇到迭代器區塊的結束點時,將會發生如下情況。
n 枚舉器物件的狀態改為after。
n MoveNext方法對呼叫方傳回false,表示迭代已經完成。
l 當一個異常被拋出並被傳播到迭代器塊之外時,將會發生如下情況。
n 在迭代器區塊之內將會因為異常傳播(exception propagation)而執行合適的finally區塊。
n 枚舉器物件的狀態改為after。
n 對於MoveNext方法的呼叫方來說,異常傳播將會繼續。
22.2.2 Current屬性
枚舉器物件的Current屬性受到迭代器區塊的yield return 語句的影響。
當枚舉器物件處於suspended狀態時,Current的值就是最後一次呼叫MoveNext時被設定的值。當枚舉器物件處於before、running或after狀態時,存取Current的所得結果是未指定的。
對於一個具有非object類型的yield 類型迭代器區塊,透過枚舉器物件的IEnumerable實現存取Current所得實現,對應於透過枚舉器物件的IEnumerator
22.2.3 Dispose方法
Dispose方法透過將枚舉器物件的狀態置為after,以清理迭代結果。
l 如果枚舉器物件的狀態是before,呼叫Dispose將改變其狀態為after。
l 如果枚舉器物件的狀態是running,呼叫Dispose的結果是為指定的。
l 如果枚舉器物件的狀態是suspended,呼叫Dispose將
n 改變其狀態為running。
n 執行finally區塊,就好像最後執行的yield return語句是一個yield break語句。如果這裡引發一個異常被拋出並傳播到迭代器體之外,枚舉器物件的狀態將被置為after,並且該異常將被傳播給Dispose方法的呼叫方。
n 改變其狀態為after。
l 如果枚舉器物件的狀態為after,呼叫Dispose沒有效果。
22.3可枚舉物件
當傳回一個可枚舉介面類型的函數成員使用迭代器區塊實作時,呼叫函數成員不會立即執行迭代器區塊程式碼。相反,一個可枚舉物件([/b]enumerable object[/b])[/b]將被建立並傳回。可枚舉物件的GetEnumerator方法傳回一個枚舉器對象,它封裝了在迭代器區塊中指定的程式碼,當枚舉器物件的MoveNext方法被呼叫時,將觸發迭代器區塊程式碼的執行。可枚舉物件具有下列特徵。
l 它實作了IEnumerable和IEnumerable
l 它使用實參值的拷貝進行初始化(如果有的話),並將實例值傳遞給函數成員。
可枚物件通常是一個由編譯器產生的可枚舉類別的實例,該類別封裝了迭代器區塊的程式碼,並實現了可枚舉接口,但其他實作方法也是可以的。如果可枚舉類別由編譯器生成,該類別將內嵌在包含函數成員的類別中,並具有私有可存取性,以及一個為編譯器所保留使用的名字(§2.4.2)。
可枚物件可以實現比在此說明的更多介面。特別的是,可枚舉物件也可以實現IEnumerator和IEnumerator
22.3.1 GetEnumerator方法
可枚舉物件提供了IEnumerable和IEnumerable
枚舉器物件使用實參值進行初始化,當可枚舉物件被初始化時其實例值將被保存,另一方面,枚舉器物件函數將如§22.2所描述。
(To be continued)
以上就是C# 2.0 Specification(迭代器)(一)的內容,更多相關內容請關注PHP中文網(www.php.cn)!