// Knife.cs.cs

namespace CustomInterfaces

{

  class Knife : IPointy

  {

    public byte Points => 1;

  }

}

После определения типов PitchFork, Fork и Knife можно определить массив объектов, совместимых с IPointy. Поскольку все элементы поддерживают один и тот же интерфейс, допускается выполнять проход по массиву и интерпретировать каждый его элемент как объект, совместимый с IPointy, несмотря на разнородность иерархий классов:

...

// Этот массив может содержать только типы,

// которые реализуют интерфейс IPointy.

IPointy[] myPointyObjects = {new Hexagon(), new Knife(),

  new Triangle(), new Fork(), new PitchFork()};

foreach(IPointy i in myPointyObjects)

{

  Console.WriteLine("Object has {0} points.", i.Points);

}

Console.ReadLine();

Просто чтобы подчеркнуть важность продемонстрированного примера, запомните, что массив заданного интерфейсного типа может содержать элементы любых классов или структур, реализующих этот интерфейс.

<p id="AutBody_Root333">Автоматическая реализация интерфейсов</p>

Хотя программирование на основе интерфейсов является мощным приемом, реализация интерфейсов может быть сопряжена с довольно большим объемом клавиатурного ввода. Учитывая, что интерфейсы являются именованными наборами абстрактных членов, для каждого метода интерфейса в каждом типе, который поддерживает данное поведение, потребуется вводить определение и реализацию. Следовательно, если вы хотите поддерживать интерфейс, который определяет пять методов и три свойства, тогда придется принять во внимание все восемь членов (иначе возникнут ошибки на этапе компиляции).

К счастью, в Visual Studio и Visual Studio Code поддерживаются разнообразные инструменты, упрощающие задачу реализации интерфейсов. В качестве примера вставьте в текущий проект еще один класс по имени PointyTestClass. Когда вы добавите к типу класса интерфейс, такой как IPointy (или любой другой подходящий интерфейс), то заметите, что по окончании ввода имени интерфейса (или при наведении на него курсора мыши в окне редактора кода) в Visual Studio и Visual Studio Code появляется значок с изображением лампочки (его также можно отобразить с помощью комбинации клавиш <Ctrl+.>). Щелчок на значке с изображением лампочки приводит к отображению раскрывающегося списка, который позволяет реализовать интерфейс (рис. 8.4 и 8.5).

Обратите внимание, что в списке предлагаются два пункта, из которых второй (явная реализация интерфейса) обсуждается в следующем разделе. Для начала выберите первый пункт. Среда Visual Studio/Visual Studio Code сгенерирует код заглушки, подлежащий обновлению (как видите, стандартная реализация генерирует исключение System.NotImplementedException, что вполне очевидно можно удалить):

namespace CustomInterfaces

{

  class PointyTestClass : IPointy

  {

    public byte Points => throw new NotImplementedException();

  }

}

На заметку! Среда Visual Studio/Visual Studio Code также поддерживает рефакторинг в форме извлечения интерфейса (Extract Interface), доступный через пункт Extract Interface (Извлечь интерфейс) меню Quick Actions (Быстрые действия). Такой рефакторинг позволяет извлечь новое определение интерфейса из существующего определения класса. Например, вы можете находиться где-то на полпути к завершению написания класса, но вдруг осознаете, что данное поведение можно обобщить в виде интерфейса (открывая возможность для альтернативных реализаций).

<p id="AutBody_Root334">Явная реализация интерфейсов</p>

Как было показано ранее в главе, класс или структура может реализовывать любое количество интерфейсов. С учетом этого всегда существует возможность реализации интерфейсов, которые содержат члены с идентичными именами, из-за чего придется устранять конфликты имен. Чтобы проиллюстрировать разнообразные способы решения данной проблемы, создайте новый проект консольного приложения по имени InterfaceNameClash и добавьте в него три специальных интерфейса, представляющих различные места, в которых реализующий их тип может визуализировать свой вывод:

namespace InterfaceNameClash

{

  // Вывести изображение на форму.

  public interface IDrawToForm

  {

    void Draw();

  }

}

namespace InterfaceNameClash

{

  // Вывести изображение в буфер памяти.

  public interface IDrawToMemory

  {

    void Draw();

  }

}

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

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