Console.WriteLine("Person number: {0}", i);  // номер лица

  Console.WriteLine("Name: {0} {1}",

    myPeople[i].FirstName, myPeople[i].LastName);  // имя и фамилия

  Console.WriteLine("Age: {0}", myPeople[i].Age);  // возраст

  Console.WriteLine();

}

Как видите, индексаторы позволяют манипулировать внутренней коллекцией подобъектов подобно стандартному массиву. Но тут возникает серьезный вопрос: каким образом сконфигурировать класс PersonCollection (или любой другой класс либо структуру) для поддержки такой функциональности? Индексатор представлен как слегка видоизмененное определение свойства С#. В своей простейшей форме индексатор создается с применением синтаксиса this[]. Ниже показано необходимое обновление класса PersonCollection:

using System.Collections;

namespace SimpleIndexer

{

  // Добавить индексатор к существующему определению класса.

  public class PersonCollection : IEnumerable

  {

    private ArrayList arPeople = new ArrayList();

    ...

    // Специальный индексатор для этого класса.

    public Person this[int index]

    {

      get => (Person)arPeople[index];

      set => arPeople.Insert(index, value);

    }

  }

}

Если не считать факт использования ключевого слова this с квадратными скобками, то индексатор похож на объявление любого другого свойства С#. Например, роль области get заключается в возвращении корректного объекта вызывающему коду. Здесь мы достигаем цели делегированием запроса к индексатору объекта ArrayList, т.к. данный класс также поддерживает индексатор. Область set контролирует добавление новых объектов Person, что достигается вызовом метода Insert() объекта ArrayList.

Индексаторы являются еще одной формой "синтаксического сахара", учитывая то, что такую же функциональность можно получить с применением "нормальных" открытых методов наподобие AddPerson() или GetPerson(). Тем не менее, поддержка индексаторных методов в специальных типах коллекций обеспечивает хорошую интеграцию с инфраструктурой библиотек базовых классов .NET Core.

Несмотря на то что создание индексаторных методов является вполне обычным явлением при построении специальных коллекций, не забывайте, что обобщенные типы предлагают такую функциональность в готовом виде. В следующем методе используется обобщенный список List объектов Person. Обратите внимание, что индексатор List можно просто применять непосредственно:

using System.Collections.Generic;

static void UseGenericListOfPeople()

{

  List myPeople = new List();

  myPeople.Add(new Person("Lisa", "Simpson", 9));

  myPeople.Add(new Person("Bart", "Simpson", 7));

  // Изменить первый объект лица с помощью индексатора.

  myPeople[0] = new Person("Maggie", "Simpson", 2);

  // Получить и отобразить каждый элемент, используя индексатор.

  for (int i = 0; i < myPeople.Count; i++)

  {

    Console.WriteLine("Person number: {0}", i);

    Console.WriteLine("Name: {0} {1}",

                       myPeople[i].FirstName, myPeople[i].LastName);

    Console.WriteLine("Age: {0}", myPeople[i].Age);

    Console.WriteLine();

  }

}

<p id="AutBody_Root406">Индексация данных с использованием строковых значений</p>

В текущей версии класса PersonCollection определен индексатор, позволяющий вызывающему коду идентифицировать элементы с применением числовых значений. Однако вы должны понимать, что это не требование индексаторного метода. Предположим, что вы предпочитаете хранить объекты Person, используя тип System.Collections.Generic.Dictionary, а не ArrayList. Поскольку типы Dictionary разрешают доступ к содержащимся внутри них элементам с применением ключа (такого как фамилия лица), индексатор можно было бы определить следующим образом:

using System.Collections;

using System.Collections.Generic;

namespace SimpleIndexer

{

Перейти на страницу:

Похожие книги