}
Во-первых, обратите внимание на то, что параметры и возвращаемое значение определяемого здесь метода Invoke() соответствуют определению делегата BinaryOp. Первые параметры членов BeginInvoke() (в данном случае это два целых числа) тоже соответствуют определению делегата BinaryOp, однако BeginInvoke() всегда имеет еще два параметра (типа AsyncCallback и object), которые используются для асинхронного вызова методов. Наконец, возвращаемое значение метода EndInvoke() тоже соответствует исходной декларации делегата, а единственным параметром метода является объект, реализующий интерфейс IAsyncResult.
Рассмотрим еще один пример. Предположим, что мы определили тип делегата, который позволяет указать на любой метод, возвращающий строку и имеющий три входных параметра System.Boolean.
public delegate string MyDelegate(bool a, bool b, bool c);
На этот раз автоматически генерируемый класс выглядит так.
sealed class MyDelegate: System.MulticastDelegate {
public MyDelegate(object target, uint functionAddress);
public stringInvoke( bool a, bool b, bool c);
public IAsyncResult BeginInvoke( bool a, bool b, bool c,AsyncCallback cb, object state);
public stringEndlnvoke(IAsyncResult result);
}
Делегаты могут также "указывать" на методы, содержащие любое число параметров out или ref. В качестве примера рассмотрим следующий тип делегата.
public delegate string MyOtherDelegate (outbool a, refbool b, int c);
Сигнатуры методов Invoke() и BeginInvoke() выглядят так, как и ожидается, но обратите внимание на метод EndInvoke(), который теперь включает и все аргументы out/ref, определенные типом делегата.
sealed class MyOtherDelegate: System.MulticastDelegate {
public MyOtherDelegate(object target, uint functionAddress);
public stringInvoke( outbool a, refbool b, int c);
public IAsyncResult BeginInvoke( outbool a, refbool b, int c, AsyncCallback cb, object state);
public stringEndInvoke (outbool a, refbool b, IAsyncResult result);
}
Итак, в результате определения делегата в C# компилятор генерирует изолированный класс с тремя методами, для которых возвращаемые значения и типы параметров соответствуют декларации делегата. Следующий псевдокод приближенно описывает соответствующий базовый шаблон.
// Это только псевдокод!
public sealed class ИмяДелегата : System.MulticastDelegate {
public ИмяДелегата (object target, uint functionAddress);
public возвращаемоеЗначениеДелегата Invoke( всеПараметрыДелегата );
public IAsyncResult BeginInvoke( всеПараметрыДелегата , AsyncCallback cb, object state);
public возвращаемоеЗначениеДелегата EndInvoke(всеПараметрыRefOutДелегата, IAsyncResult result);
}
Базовые классы System.MulticastDelegate и System.Delegate
Таким образом, при создании типов c помощью) ключевого слова delegate в C# вы неявно объявляете тип класса, являющегося производным от System.MulticastDelegate. Этот класс обеспечивает своим потомкам доступ к списку с адресами тех методов, которые поддерживаются типом делегата, а также предлагает несколько дополнительных методов (и ряд перегруженных операций), обеспечивающих взаимодействие со списком вызовов. Вот программный код некоторых членов System.MulticastDelegate.
[Serializable]
public abstract class MulticastDelegate: Delegate {
// Методы
public sealed override Delegate[] GetInvocationList();
public static bool operator==(MulticastDelegate d1, MulticastDelegate d2);
public static bool operator!=(MulticastDelegate d1, MulticastDelegate d2);
// Поля
private IntPtr _invocationCount;
private object _invocationList;
}
Дополнительные функциональные возможности System.MulticastDelegate получает от своего родительского класса System.Delegate. Вот часть определения этого класса.
[Serializable, ClassInterface(ClassInterfaceType.AutoDual)]
public abstract class Delegate: ICloneable, ISerializable {
// Методы
public static Delegate Combine(params Delegate[] delegates);
public static Delegate Combine(Delegate a, Delegate b);
public static Delegate Remove(Delegate source, Delegate value);
public static Delegate RemoveAll(Delegate source, Delegate value);
// Перегруженные операции
public static bool operator==(Delegate d1, Delegate d2);
public static bool operator!=(Delegate d1, Delegate d2);
// Свойства
public MethodInfo Method {get;}
public object Target {get;}
}
Здесь следует подчеркнуть, что от вас никогда не потребуется непосредственно получать производные этих базовых классов, и вы можете ограничить себя использованием только членов, указанных в табл. 8.1.
Таблица 8.1.Избранные члены System.MulticastDelegate и System.Delegate
Читать дальше