В целях иллюстрации внесите в файл Program.cs показанные ниже изменения, где групповое преобразование методов применяется для регистрации и отмены регистрации подписки на уведомления:

...

Console.WriteLine("***** Method Group Conversion *****\n");

Car c2 = new Car();

// Зарегистрировать простое имя метода.

c2.RegisterWithCarEngine(OnCarEngineEvent);

Console.WriteLine("***** Speeding up *****");

for (int i = 0; i < 6; i++)

{

  c2.Accelerate(20);

}

// Отменить регистрацию простого имени метода.

c2.UnRegisterWithCarEngine(OnCarEngineEvent);

// Уведомления больше не поступают!

for (int i = 0; i < 6; i++)

{

  c2.Accelerate(20);

}

Console.ReadLine();

Обратите внимание, что мы не создаем напрямую ассоциированный объект делегата, а просто указываем метод, который соответствует ожидаемой сигнатуре делегата (в данном случае метод, возвращающий void и принимающий единственный аргумент string). Имейте в виду, что компилятор C# по-прежнему обеспечивает безопасность в отношении типов. Таким образом, если метод OnCarEngineEvent() не принимает string и не возвращает void, тогда возникнет ошибка на этапе компиляции.

<p id="AutBody_Root454">Понятие обобщенных делегатов</p>

В главе 10 упоминалось о том, что язык C# позволяет определять обобщенные типы делегатов. Например, предположим, что необходимо определить тип делегата, который может вызывать любой метод, возвращающий void и принимающий единственный параметр. Если передаваемый аргумент может изменяться, то это легко смоделировать с использованием параметра типа. Взгляните на следующий код внутри нового проекта консольного приложения по имени GenericDelegate:

Console.WriteLine("***** Generic Delegates *****\n");

// Зарегистрировать цели.

MyGenericDelegate strTarget =

  new MyGenericDelegate(StringTarget);

strTarget("Some string data");

// Использовать синтаксис группового преобразования методов

MyGenericDelegate intTarget = IntTarget;

intTarget(9);

Console.ReadLine();

static void StringTarget(string arg)

{

  Console.WriteLine("arg in uppercase is: {0}", arg.ToUpper());

}

static void IntTarget(int arg)

{

  Console.WriteLine("++arg is: {0}", ++arg);

}

// Этот обобщенный делегат может вызывать любой метод, который

// возвращает void и принимает единственный параметр типа Т.

public delegate void MyGenericDelegate(T arg);

Как видите, в типе делегата MyGenericDelegate определен единственный параметр, представляющий аргумент для передачи цели делегата. При создании экземпляра этого типа должно быть указано значение параметра типа наряду с именем метода, который делегат может вызывать. Таким образом, если указать тип string, тогда целевому методу будет отправляться строковое значение:

// Создать экземпляр MyGenericDelegate

// с указанием string в качестве параметра типа.

MyGenericDelegate strTarget = StringTarget;

strTarget("Some string data");

С учетом формата объекта strTarget метод StringTarget теперь должен принимать в качестве параметра единственную строку:

static void StringTarget(string arg)

{

  Console.WriteLine(

    "arg in uppercase is: {0}", arg.ToUpper());

}

<p id="AutBody_Root455">Обобщенные делегаты Action<> и Func<></p>

В настоящей главе вы уже видели, что когда нужно применять делегаты для обратных вызовов в приложениях, обычно должны быть выполнены описанные далее шаги.

1. Определить специальный делегат, соответствующий формату метода, на который он указывает.

2. Создать экземпляр специального делегата, передав имя метода в качестве аргумента конструктора.

3. Косвенно обратиться к методу через вызов Invoke() на объекте делегата.

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

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