static void Main() {
MyClass mc = new MyClass();
Console.WriteLine("Возвратить по очереди первые 7 букв:");
foreach (char ch in mc.MyItr(7))
Console.Write(ch + " ");
Console.WriteLine("\n");
Console.WriteLine("Возвратить по очереди буквы от F до L:");
foreach (char ch in mc.MyItr(5, 12))
Console.Write(ch + " ");
Console.WriteLine();
}
}
Эта программа дает следующий результат.
Возвратить по очереди первые 7 букв:
А В С D Е F G
Возвратить по очереди буквы от F до L:
F G Н I J К L
Создание обобщенного итератора
В приведенных выше примерах применялись необобщенные итераторы, но, конечно, ничто не мешает создать обобщенные итераторы. Для этого достаточно возвратить объект обобщенного типа IEnumeratorили IEnumerable. Ниже приведен пример создания обобщенного итератора.
// Простой пример обобщенного итератора,
using System;
using System.Collections.Generic;
class MyClass {
T[] array;
public MyClass(T[] a) {
array = a;
}
// Этот итератор возвращает символы из массива chrs.
public IEnumerator GetEnumerator() {
foreach (T obj in array)
yield return obj;
}
}
class GenericItrDemo {
static void Main() {
int[] nums = { 4, 3, 6, 4, 7, 9 };
MyClass mc = new MyClass(nums);
foreach (int x in mc)
Console.Write(x + " ");
Console.WriteLine();
bool[] bVals = { true, true, false, true };
MyClass mc2 = new MyClass(bVals);
foreach (bool b in mc2)
Console.Write(b + " ");
Console.WriteLine();
}
}
Вот к какому результату приводит выполнение этой программы.
4 3 6 4 7 9
True True False True
В данном примере массив, состоящий из возвращаемых по очереди объектов, передается конструктору класса MyClass. Тип этого массива указывает в качестве аргумента типа в конструкторе класса MyClass.
Метод GetEnumerator()оперирует данными обобщенного типа Т и возвращает перечислитель типа IEnumerator. Следовательно, итератор, определенный в классе MyClass, способен перечислять данные любого типа.
В С# имеется специальное средство, называемое инициализатором коллекции и упрощающее инициализацию некоторых коллекций. Вместо того чтобы явно вызывать метод Add(), при создании коллекции можно указать список инициализаторов. После этого компилятор организует автоматические вызовы метода Add(), используя значения из этого списка. Синтаксис в данном случае ничем не отличается от инициализации массива. Обратимся к следующему примеру, в котором создается коллекция типа List, инициализируемая символами С, А, Е, В, D и F.
List lst = new List() { 'С', 'А', 'Е', 'В', 'D', 'F' };
После выполнения этого оператора значение свойства lst.Countбудет равно 6, поскольку именно таково число инициализаторов. А после выполнения следующего цикла foreach:
foreach(ch in lst)
Console.Write(ch + " ");
получится такой результат:
С A E В D F
Для инициализации коллекции типа LinkedList, в которой хранятся пары "ключ-значение", инициализаторы приходится предоставлять парами, как показано ниже.
SortedListcint, string> lst =
new SortedList()
{ {1, "один"}, {2, "два" }, {3, "три"} };
Компилятор передаст каждую группу значений в качестве аргументов методу Add(). Следовательно, первая пара инициализаторов преобразуется компилятором в вызов Add(1, "один").
Компилятор вызывает метод Add()автоматически для ввода инициализаторов в коллекцию, и поэтому инициализаторы коллекций можно использовать только в коллекциях, поддерживающих открытую реализацию метода Add(). Это означает, что инициализаторы коллекций нельзя использовать в коллекциях типа Stack, Stack, Queueили Queue, поскольку в них метод Add()не поддерживается. Их нельзя применять также в тех коллекциях типа LinkedList, где метод Add()предоставляется как результат явной реализации соответствующего интерфейса.
Читать дальше