• 欢迎访问我爱CSharp学习网,这里有最新最全的C#书籍,C#视频。
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏我爱C#学习网吧
  • 推荐使用最新版Chrome浏览器和火狐浏览器访问本网站

C# 知识回顾 – 委托 delegate

C#杂烩 52csharp 500次浏览 0个评论 扫描二维码

(点击上方蓝字,可快速关注我们)


来源:反骨仔(二五仔)

链接:cnblogs.com/liqingwen/p/6031892.html


What’s 委托


delegate 是表示对具有特定参数列表和返回类型的方法的引用的类型。在实例化委托时,你可以将其实例与任何具有兼容签名和返回类型的方法相关联。


你可以通过委托实例调用方法。委托用于将方法作为参数传递给其他方法。事件处理程序就是通过委托调用的方法。你可以创建一个自定义方法,当发生特定事件时,某个类(如 Windows 控件)就可以调用你的方法。


下面的示例演示了一个委托声明:


public delegate int Del(int x, int y);


可将任何可访问类或结构中与委托类型匹配的任何方法分配给委托。该方法可以是静态方法,也可以是实例方法。这样便能通过编程方式来更改方法调用,还可以向现有类中插入新代码。


【备注】在方法重载的上下文中,方法的签名不包括返回值。但在委托的上下文中,签名包括返回值。换句话说,方法和委托必须具有相同的返回类型。


将方法作为参数进行引用的能力使委托成为定义回调方法的理想选择。例如,对比较两个对象的方法的引用可以作为参数传递到排序算法中。由于比较代码在一个单独的过程中,因此可通过更常见的方式编写排序算法。


委托的属性概述


  • 类似于 C 和 C++ 中的函数指针,但它们是类型安全的。


  • 允许将方法作为参数进行传递。


  • 可用于定义回调方法。


  • 委托可以链接在一起;例如,可以对一个事件调用多个方法。


  • 方法不必与委托类型完全匹配。


使用委托


委托是安全封装方法的类型,类似于 C 和 C++ 中的函数指针。与 C 函数指针不同的是,委托是面向对象的、类型安全的和可靠的。委托的类型由委托的名称确定。


//该委托可以封装 “,参数类型 string,返回类型 void” 的方法 

public delegate void MyDel(string message);

 

委托对象通常通过提供委托将封装的方法的名称或使用匿名方法构造。对委托进行实例化后,委托会将对其进行的方法调用传递到该方法。调用方传递到委托的参数将传递到该方法,并且委托会将方法的返回值(如果有)返回到调用方。这被称为调用委托。实例化的委托可以按封装的方法本身进行调用。例如:


//该委托名为 MyDel,可以封装 “参数类型 string,返回值类型 void” 的方法 

public delegate void MyDel(string message);

class Program

{

        static void Main(string[] args)

        {

            //实例化委托

            MyDel del = Print;

            //调用委托

            del(“Hi”);

            Console.Read();

        }

        /// <summary>

        /// 打印文本

        /// </summary>

        /// <remarks>这是一个可用于 MyDel 委托的方法</remarks>

        /// <param name=”message”></param>

        private static void Print(string message)

        {

            Console.WriteLine(message);

        }

    }


C# 知识回顾 - 委托 delegate


委托类型派生自 .NET Framework 中的 Delegate 类。委托类型是封装的,它们不能派生出其他类,也不能从 Delegate 派生出自定义类。由于实例化的委托是一个对象,因此可以作为参数传递或分配给一个属性。


这允许方法作为参数接受委托并在稍后调用委托。这被称为异步回调,是在长进程完成时通知调用方的常用方法。当以这种方式使用委托时,使用委托的代码不需要知道要使用的实现方法。功能类似于封装接口提供的功能。  


回调的另一个常见用途是定义自定义比较方法并将该委托传递到短方法。  它允许调用方的代码成为排序算法的一部分。  以下示例方法使用 Del 类型作为参数:


class Program

{

        static void Main(string[] args)

        {

            MyDel del = Print;

            CallbackMethod(100, 150, del);  //将委托传递到 CallbackMethod 方法

            Console.Read();

        }

        /// <summary>

        /// 回调方法

        /// </summary>

        /// <param name=”m”></param>

        /// <param name=”n”></param>

        /// <param name=”del”></param>

        private static void CallbackMethod(int m, int n, MyDel del)

        {

            del((m + n).ToString());

        }

        private static void Print(string message)

        {

            Console.WriteLine(message);

        }

    }


C# 知识回顾 - 委托 delegate


CallbackMethod 的作用是将字符串传递到其他方法。由于该方法可以使用任意数量的参数,此功能特别强大。 


当委托构造为封装实例方法时,委托将同时引用实例和方法。委托不知道除其所封装方法以外的实例类型,因此委托可以引用任何类型的对象,只要该对象上有与委托签名匹配的方法。当委托构造为封装静态方法时,委托仅引用方法。  请考虑以下声明:  


//该委托可以封装 “名 MyDel,参数类型 string,返回值类型 void” 的方法 

public delegate void MyDel(string message);

class MyClass

{

        public void Print1(string message)

        {

            Console.WriteLine($”{message} – {nameof(Print1)}”);

        }

        public void Print2(string message)

        {

            Console.WriteLine($”{message} – {nameof(Print2)}”);

        }

    }

class Program

{

        static void Main(string[] args)

        {

            var myClass = new MyClass();

            MyDel del1 = myClass.Print1;

            MyDel del2 = myClass.Print2;

            MyDel del3 = Print;

            var del = del1 + del2;

            del += del3;    //这里使用 +=

            del(“Hi!”);

            Console.Read();

        }

        private static void Print(string message)

        {

            Console.WriteLine($”{message} – {nameof(Print)}”);

        }

    }


C# 知识回顾 - 委托 delegate


调用时,委托可以调用多个方法。这被称为多播。若要向委托的方法列表(调用列表)添加其他方法,只需使用加法运算符或加法赋值运算符(“+”或“+=”)添加两个委托。


此时,del 的调用列表中包含三个方法,分别为 Print1、Print2 和 Print。原有的三个委托(del1、del2 和 del3)保持不变。调用 allMethodsDelegate 时,将按顺序调用所有三个方法。如果委托使用引用参数,引用将按相反的顺序传递到所有这三个方法,并且一种方法进行的任何更改都将在另一种方法上见到。


当方法引发未在方法内捕获到的异常时,该异常将传递到委托的调用方,并且不会调用调用列表中的后续方法。如果委托具有返回值和/或输出参数,它将返回上次调用方法的返回值和参数。若要删除调用列表中的方法,请使用减法运算符或减法赋值运算符(“-”或“-=”)。  例如:  


static void Main(string[] args)

{

            var myClass = new MyClass();

            MyDel del1 = myClass.Print1;

            MyDel del2 = myClass.Print2;

            MyDel del3 = Print;

            var del = del1 + del2;

            del += del3;    //使用 +=

            del(“Hi!”);

            Console.WriteLine(“======分割线======”);

            del -= del2;    //使用 -=

            del(“Hi!”);

            Console.Read();

}


C# 知识回顾 - 委托 delegate


由于委托类型派生自 System.Delegate,因此可以在委托上调用该类定义的方法和属性。例如,若要查询委托调用列表中方法的数量,你可以编写:


static void Main(string[] args)

{

            var myClass = new MyClass();

            MyDel del1 = myClass.Print1;

            MyDel del2 = myClass.Print2;

            MyDel del3 = Print;

            var del = del1 + del2;

            del += del3;    //使用 +=

            //del(“Hi!”);

            var count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量

            Console.WriteLine(count);

            Console.WriteLine(“======分割线======”);

            del -= del2;    //使用 -=

            //del(“Hi!”);

            count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量

            Console.WriteLine(count);

            Console.Read();

}


C# 知识回顾 - 委托 delegate


调用列表中具有多个方法的委托派生自 MulticastDelegate,该类属于 System.Delegate 的子类。由于这两个类都支持 GetInvocationList,因此在其他情况下,上述代码也将产生作用。 


多播委托广泛用于事件处理中。事件源对象将事件通知发送到已注册接收该事件的接收方对象。


若要注册一个事件,接收方需要创建用于处理该事件的方法,然后为该方法创建委托并将委托传递到事件源。


事件发生时,源调用委托。然后,委托将对接收方调用事件处理方法,从而提供事件数据。给定事件的委托类型由事件源确定。  


 

C# 知识回顾 - 委托 delegate


我爱CSharp学习网 , 版权所有丨如未注明 , 均为原创丨本网站采用BY-NC-SA协议进行授权 , 转载请注明C# 知识回顾 – 委托 delegate
喜欢 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址