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