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()
предоставляется как результат явной реализации соответствующего интерфейса.
Читать дальше