Г. Шилдт - Полный справочник по C# (1160789), страница 76
Текст из файла (страница 76)
Таким образом, создавая делегат, вы по сути создаете объект, который может содержать ссылку на метод. Более того, этот метод можно вызватьпосредством соответствующей ссылки. Таким образом, делегат может вызывать метод,на который он ссылается.На первый взгляд идея ссылки на метод может показаться странной, посколькуобычно мы имеем дело с ссылками, которые указывают на объекты, но в действительности здесь разница небольшая. Как разъяснялось выше, ссылка по существупредставляет собой адрес памяти. Следовательно, ссылка на объект — это адрес объекта. Даже несмотря на то что метод не является объектом, он тоже имеет отношениек физической области памяти, а адрес его точки входа — это адрес, к которому происходит обращение при вызове метода.
Этот адрес можно присвоить делегату. Если ужделегат ссылается на метод, этот метод можно вызвать посредством данного делегата.11 la заметкуЕсли вы знакомы с C/C++, то вам будет полезно узнать, что делегааналогичен указателю на функцию в C/C++Важно понимать, что во время выполнения программы один и тот же делегатможно использовать для вызова различных методов, просто заменив метод, на который ссылается этот делегат. Таким образом, метод, который будет вызван делегатом,определяется не в период компиляции программы, а во время ее работы. В этом и состоит достоинство делегата.Делегат объявляется с помощью ключевого слова d e l e g a t e .
Общая форма объявления делегата имеет следующий вид:d e l e g a t e тип_возвратаимя{список_параметров);Здесь элемент тип_возврата представляет собой тип значений, возвращаемых методами, которые этот делегат будет вызывать. Имя делегата указывается элементомимя. Параметры, принимаемые методами, которые вызываются посредством делегата,задаются с помощью элемента список_параметров. Делегат может вызывать толькотакие методы, у которых тип возвращаемого значения и список параметров (т.е.
егосигнатура) совпадают с соответствующими элементами объявления делегата.Делегат может вызывать либо метод экземпляра класса, связанный с объектом, илистатический метод, связанный с классом.Чтобы увидеть делегат в действии, начнем со следующего простого примера:// Простой пример использования делегата.using System;// Объявляем делегат.delegate string strMod(string stx);class DelegateTest {// Метод заменяет пробелы дефисами.410Часть I.
Язык С#static string replaceSpaces(string a) {Console.WriteLine("Замена пробелов дефисами.");return a.Replace(' ', ' - ' ) ;// Метод удаляет пробелы.static string removeSpaces(string a) {string temp = "";int i;Console.WriteLine("Удаление пробелов.");for(i=0; i < a.Length; i++)if (a[i] != ' f ) temp += a[i] ;return temp;// Метод реверсирует строку,static string reverse(string a) {string temp = "";int i, j;Console.WriteLine("Реверсирование строки.");for(j=0, i=a.Length-l; i >= 0; i — ,temp += a[i];return temp;public static void Main() {// Создание делегата.strMod strOp = new strMod(replaceSpaces);string str;// Вызываем методы посредством делегата.str = str0p("3TO простой тест.");Console.WriteLine("Результирующая строка: " + str);Console.WriteLine();strOp = new strMod(removeSpaces);str = strOp("Это простой тест.");Console.WriteLine("Результирующая строка: " + str);Console.WriteLine();strOp = new strMod(reverse);str = strOp("Это простой тест.");Console.WriteLine("Результирующая строка: " + str);Результаты выполнения этой программы выглядят так:Замена пробелов дефисами.Результирующая строка: Это-простой-тест.Удаление пробелов.Результирующая строка: Этопростойтест.Реверсирование строки.Результирующая строка: .тсет йотсорп отЭГлава 15.
Делегаты и события411Итак, в программе объявляется делегат с именем strMod, который принимаетодин параметр типа s t r i n g и возвращает string-значение. В классе DelegateTestобъявляны три статических метода, сигнатура которых совпадает с сигнатурой, заданной делегатом. Эти методы предназначены для модификации строк определенноговида. Обратите внимание на то, что метод replaceSpaces () для замены пробеловдефисами использует метод Replace () — один из методов класса s t r i n g .В методе Main () создается ссылка типа strMod с именем strOp, и ей присваивается ссылка на метод replaceSpaces (). Внимательно рассмотрите следующую строку:1 strMod strOp = new s t r M o d ( r e p l a c e S p a c e s ) ;Обратите внимание на то, что метод replaceSpaces () передается делегату в качестве параметра. Здесь используется только имя метода (параметры не указываются).Это наблюдение можно обобщить: при реализации делегата задается только имя метода, на который должен ссылаться этот делегат.
Кроме того, объявление метода должносоответствовать объявлению делегата. В противном случае вы получите сообщение обошибке еще во время компиляции.Затем метод replaceSpaces () вызывается посредством экземпляра делегата сименем strOp, как показано в следующей строке:I s t r = strOp("Это простой т е с т . " ) ;Поскольку экземпляр strOp ссылается на метод replaceSpaces ( ) , то вызываетсяименно метод replaceSpaces (). Затем экземпляру делегата strOp присваиваетсяссылка на метод removeSpaces (), после чего strOp вызывается снова. На этот развызывается метод removeSpaces ().Наконец, экземпляру делегата strOp присваивается ссылка на метод r e v e r s e (), иstrOp вызывается еще раз.
Это, как нетрудно догадаться, приводит к вызову методаreverse().Главное в этом примере то, что вызов экземпляра делегата s t r o p трансформируется в обращение к методу, на который ссылается strOp при вызове. Таким образом,решение о вызываемом методе принимается во время выполнения программы, а не впериод компиляции.Несмотря на то что в предыдущем примере используются статические методы, делегат может также ссылаться на методы экземпляров класса. Однако он должен приэтом использовать объектную ссылку. Например, вот как выглядит предыдущая программа, переписанная с целью инкапсуляции операций над строками внутри классаStringOps:// Делегаты могут ссылаться также на методы// экземпляров класса.using System;// Объявляем делегат.delegate string strMod(string str);class StringOps {// Метод заменяет пробелы дефисами,public string replaceSpaces(string a) {Console.WriteLine("замена пробелов дефисами.");freturn a.Replace( ', ' - ' ) ;}// Метод удаляет пробелы.public string removeSpaces(string a) {string temp = "";int i;412Часть I.
Язык С#Console.WriteLine("Удаление пробелов. ") ;for(i=0; i < a.Length; i++)if(a[i] != f ') temp += a[i];return temp;// Метод реверсирует строку,public string reverse(string a) {string temp = "";int i f j;Console.WriteLine("Реверсирование строки.")for(j=0, i=a.Length-l; i >= 0; i — ,temp += a[i];return temp;class DelegateTest {public static void Main() {StringOps so = new StringOps(); // Создаем экземпляр// класса StringOps.// Создаем делегат.strMod strOp = new strMod(so.replaceSpaces);string str;// Вызываем методы с использованием делегатов,str = str0p("3TO простой тест.");Console.WriteLine("Результирующая строка: " + str);Console.WriteLine();strOp = new strMod(so.removeSpaces);str = strOp("Это простой тест.");Console.WriteLine("Результирующая строка: " + str);Console.WriteLine();strOp = new strMod(so.reverse);str = strOp("Это простой тест.");Console.WriteLine("Результирующая строка: " + str);Результаты выполнения этой программы совпадают с результатами предыдущейверсии, но в этом случае делегат ссылается на методы экземпляра класса StringOps.Многоадресатная передачаОдна из самых интересных возможностей делегата — поддержка многоадресатнойпередачи (multicasting).
Выражаясь простым языком, Многоадресатная передача — этоспособность создавать список вызовов (или цепочку вызовов) методов, которые должны автоматически вызываться при вызове делегата. Такую цепочку создать нетрудно.Достаточно создать экземпляр делегата, а затем для добавления методов в эту цепочкуиспользовать оператор "+=". Для удаления метода из цепочки используется оператор" - = " . (Можно также для добавления и удаления методов использовать в отдельностиоператоры " + " , " - " и " = " , но чаще применяются составные операторы "+=" и "-=".)Глава 15.
Делегаты и события413Делегат с многоадресатной передачей имеет одно ограничение: он должен возвращатьтип void.Рассмотрим следующий пример многоадресатной передачи. Это — переработанный вариант предыдущих примеров, в котором тип s t r i n g для значений, возвращаемых методами обработки строк, заменен типом void, а для возврата модифицированных строк используется ref-параметр.// Демонстрация использования многоадресатной передачи.using System;// Объявляем делегат.delegate void strMod(ref string str);class StringOps {// Метод заменяет пробелы дефисами,static void replaceSpaces(ref string a) {Console.WriteLine("Замена пробелов дефисами.");a = a.Replace(' •, f - f ) ;}// Метод удаляет пробелы.static void removeSpaces(ref string a) {string temp = "";int i;Console.WriteLine("Удаление пробелов.");for(i=0; i < a.Length; i++)if(a[i] != f f ) temp += a[i];a = temp;}// Метод реверсирует строку,static void reverse(ref string a) {string temp = "";int i, j;Console.WriteLine("Реверсирование строки.");for(j=0, i=a.Length-l; i >= 0; i — , j++)temp += a[i];a = temp;}public static void Main() {// Создаем экземпляры делегатов.strMod strOp;strMod replaceSp = new strMod(replaceSpaces);strMod removeSp = new strMod(removeSpaces);strMod reverseStr = new strMod(reverse);string str = "Это простой тест.";// Организация многоадресатной передачи.strOp = replaceSp;strOp += reverseStr;// Вызов делегата с многоадресатной передачей.strOp(ref str) ;414Часть I.