Рассмотрим эту программу более подробно. Во-первых, обратите внимание на то, что в классеMyClassне указываетсяIEnumeratorв качестве реализуемого интерфейса. При создании итератора компилятор реализует этот интерфейс автоматически. И во-вторых, обратите особое внимание на методGetEnumerator (), который ради удобства приводится ниже еще раз.
// Этот итератор возвращает символы из массива chrs. public IEnumerator GetEnumerator() {
foreach(char ch in chrs) yield return ch;
}
Это и есть итератор для объектов классаMyClass.Как видите, в нем явно реализуется методGetEnumerator (), определенный в интерфейсеIEnumerable.А теперь перейдем непосредственно к телу данного метода. Оно состоит из циклаforeach,в котором возвращаются элементы из массиваchrs.И делается это с помощью оператораyield return.Этот оператор возвращает следующий объект в коллекции, которым в данном случае оказывается очередной символ в массивеchrs.Благодаря этому средству обращение к объектутстипаMyClassорганизуется в циклеforeachвнутри методаMain ().
Обозначениеyieldслужит в языке C# в качестве
Следует особо подчеркнуть, что итератор не обязательно должен опираться на массив или коллекцию другого типа. Он должен просто возвращать следующий элемент из совокупности элементов. Это означает, что элементы могут быть построены динамически с помощью соответствующего алгоритма. В качестве примера ниже приведена версия предыдущей программы, в которой возвращаются все буквы английского алфавита, набранные в верхнем регистре. Вместо массива буквы формируются в циклеfor.
// Пример динамического построения значений,
// возвращаемых по очереди с помощью итератора.
using System;
using System.Collections;
class MyClass { char ch = fAf;
// Этот итератор возвращает буквы английского // алфавита, набранные в верхнем регистре.
public IEnumerator GetEnumerator() {
for(int i=0; i < 26; i++)
yield return (char) (ch + i) ;
}
}
class ItrDemo2 {
static void Main() {
MyClass me = new MyClass();
foreach(char ch in me)
Console.Write(ch + " ");
Console.WriteLine();
}
}
Вот к какому результату приводит выполнение этой программы.
ABCDEFGHI JKLMNOPQRSTUVWXYZ
Прерывание итератора
Для преждевременного прерывания итератора служит следующая форма оператора yield.
yield break;