string str = "Это простой тест.";
// Организовать групповую адресацию.
strOp = replaceSp;
strOp += reverseStr;
// Обратиться к делегату с групповой адресацией.
strOp(ref str);
Console.WriteLine("Результирующая строка: " + str);
Console.WriteLine();
// Удалить метод замены пробелов и добавить метод удаления пробелов.
strOp -= replaceSp;
strOp += removeSp;
str = "Это простой тест."; // восстановить исходную строку
// Обратиться к делегату с групповой адресацией.
strOp(ref str);
Console.WriteLine("Результирующая строка: " + str);
Console.WriteLine() ;
}
}
Выполнение этого кода приводит к следующему результату.
Замена пробелов дефисами.
Обращение строки.
Результирующая строка: .тсет-йотсорп-отЭ
Обращение строки.
Удаление пробелов.
Результирующая строка: .тсетйотсорпотЭ
В методе Main()
из рассматриваемого здесь примера кода создаются четыре экземпляра делегата. Первый из них, strOp
, является пустым, а три остальных ссылаются на конкретные методы видоизменения строки. Затем организуется групповая адресация для вызова методов RemoveSpaces()
и Reverse()
. Это делается в приведенных ниже строках кода.
strOp = replaceSp;
strOp += reverseStr
Сначала делегату strOp
присваивается ссылка replaceSp
, а затем с помощью оператора +=
добавляется ссылка reverseStr
. При обращении к делегату strOp
вызываются оба метода, заменяя пробелы дефисами и обращая строку, как и показывает приведенный выше результат.
Далее ссылка replaceSp
удаляется из цепочки вызовов в следующей строке кода:
strOp -= replaceSp;
и добавляется ссылка removeSp
в строке кода.
strOp += removeSp;
После этого вновь происходит обращение к делегату strOp
. На этот раз обращается строка с удаленными пробелами.
Цепочки вызовов являются весьма эффективным механизмом, поскольку они позволяют определить ряд методов, выполняемых единым блоком. Благодаря этому улучшается структура некоторых видов кода. Кроме того, цепочки вызовов имеют особое значение для обработки событий, как станет ясно в дальнейшем.
Делегаты становятся еще более гибкими средствами программирования благодаря двум свойствам:
Ниже приведен пример, демонстрирующий ковариантность и контравариантность.
// Продемонстрировать ковариантность и контравариантность.
using System;
class X {
public int Val;
}
// Класс Y, производный от класса X.
class Y : X { }
// Этот делегат возвращает объект класса X и
// принимает объект класса Y в качестве аргумента,
delegate X ChangeIt(Y obj);
class CoContraVariance {
// Этот метод возвращает объект класса X и
// имеет объект класса X в качестве параметра,
static X IncrA(X obj) {
X temp = new X();
temp.Val = obj.Val + 1;
return temp;
}
// Этот метод возвращает объект класса Y и
// имеет объект класса Y в качестве параметра,
static Y IncrB(Y obj) {
Y temp = new Y();
temp.Val = obj.Val + 1;
return temp;
}
static void Main() {
Y Yob = new Y();