当前位置: 澳门新濠3559 > 编程 > 正文

命名方法是通过参数的形式进行传递来进行委托

时间:2019-11-29 05:50来源:编程
【博主】反骨仔 【原文】  [C#] C# 知识回顾, 序 上篇《C# 知识回顾 - 委托delegate》已经介绍委托的基本知识,这里是对其进行补充说明及加深理解。   C# 知识回顾 - 委托 delegate 【博主

【博主】反骨仔    【原文】 

[C#] C# 知识回顾,

  上篇《C# 知识回顾 - 委托 delegate》已经介绍委托的基本知识,这里是对其进行补充说明及加深理解。

 

C# 知识回顾 - 委托 delegate

【博主】反骨仔    【原文】

目录

  • 两个简单 Demo:委托的声明与调用
  • 创建多播委托
  • 委托的简单演化过程

澳门新濠3559, 

目录

  • What's 委托
  • 委托的属性概述
  • Use 委托

 

一、两个简单 Demo:委托的声明与调用

  在很久以前,委托的创建都是与命名方法直接关联,即直接通过命名方法对委托进行实例化操作,在创建时,编译器帮我们做了一些语法简化的工作,事实上在创建委托对象的时候,命名方法是通过参数的形式进行传递来进行委托对象的创建。当然,该方法不限于静态方法和实例方法。  

 1     class Program
 2     {
 3         //声明一个委托
 4         delegate void MyDel(string message);
 5 
 6         
 7         static void Main(string[] args)
 8         {
 9             //使用静态方法作为参数实例化委托
10             MyDel del = Print;
11         }
12 
13         //声明一个方法
14         private static void Print(string message)
15         {
16             Console.WriteLine(message);
17         }
18     }

  使用命名方法构造的委托可以封装静态方法或实例方法。在过去的 C# 中,命名方法是对委托进行实例化的唯一方式,而现在,我们可以使用 lambda 表达式和匿名方法了。  

  【备注】①作为委托参数传递的方法必须与委托声明具有相同的签名和返回值。②委托实例可以封装静态或实例方法。③尽管委托可以使用 out 参数,但建议不要将其用于多路广播事件委托。

 

  示例1:以下是使用委托的一个简单示例。 注意,委托 MyDel 和关联的方法 Print 具有相同的签名(即便方法的参数名称 m 和 n 的位置替换),而不是参数名相同  。

 1     class Program
 2     {
 3         //声明一个委托
 4         delegate void MyDel(int n, int m);
 5 
 6         static void Main(string[] args)
 7         {
 8             //使用静态方法 Print 作为参数实例化委托
 9             MyDel del = Print;
10             Console.WriteLine("准备好了哦,要开始调用委托了哦!");
11 
12             for (int i = 0; i < 10; i++)
13             {
14                 del(i, 1);
15             }
16 
17             Console.Read();
18         }
19 
20         //声明一个方法
21         private static void Print(int m, int n)
22         {
23             Console.Write(m - n + " ");
24         }
25     }

澳门新濠3559 1

 

   示例2:通过匿名方法实例化委托,修改示例1,结果同上。

 1     class Program
 2     {
 3         //声明一个委托
 4         delegate void MyDel(int n, int m);
 5 
 6         static void Main(string[] args)
 7         {
 8             //使用静态方法 Print 作为参数实例化委托
 9             //MyDel del = Print;
10 
11             //使用匿名方法
12             MyDel del = delegate (int m, int n)
13               {
14                   Console.Write(m - n + " ");
15               };
16             Console.WriteLine("准备好了哦,要开始调用委托了哦!");
17 
18             for (int i = 0; i < 10; i++)
19             {
20                 del(i, 1);
21             }
22 
23             Console.Read();
24         }
25     }

   【备注】感谢 2 楼 随碟附送520 的修正。感谢 4 楼 潇十一郎 的补充,也可以用 Lambda 的形式 (m, n) => 创建委托。

 

What's 委托

  delegate 是表示对具有特定参数列表和返回类型的方法的引用的类型。在实例化委托时,你可以将其实例与任何具有兼容签名和返回类型的方法相关联。你可以通过委托实例调用方法。委托用于将方法作为参数传递给其他方法。事件处理程序就是通过委托调用的方法。你可以创建一个自定义方法,当发生特定事件时,某个类(如 Windows 控件)就可以调用你的方法。

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

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

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

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

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

 

二、创建多播委托

  创建多播委托,也可以说是合并委托,你可以使用

  • 或者 += 将多个委托对象中的列表进行组合。在调用多播委托的同时,调用的顺序会依照调用列表中的顺序。需要注意的是,在合并的同时只能合并相同类型的委托。可以使用
  • 和 -= 从多播委托中移除一个方法或方法列表。

    1 class Program 2 { 3 //声明一个委托 4 delegate void MyDel(); 5 6 static void Main(string[] args) 7 { 8 //Action:你也可以自己尝试使用 Action 代替 MyDel 试试 9 10 MyDel del = Start; //创建一个委托对象 11 MyDel del2 = Stop; //创建一个委托对象 12 MyDel del3 = del + Stop; //合并前两个委托对象 13 MyDel del4 = del3 - Start; //移除一个委托对象 14 15 Console.WriteLine($"This is {nameof(del)}: "); 16 del(); 17 Console.WriteLine($"This is {nameof(del2)}: "); 18 del2(); 19 Console.WriteLine($"This is {nameof(del3)}: "); 20 del3(); 21 Console.WriteLine($"This is {nameof(del4)}: "); 22 del4(); 23 24 Console.Read(); 25 } 26 27 private static void Start() 28 { 29 Console.WriteLine($" {nameof(Start)}..."); 30 } 31 32 private static void Stop() 33 { 34 Console.WriteLine($" {nameof(Stop)}!"); 35 } 36 }

澳门新濠3559 2

 

委托的属性概述

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

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

  • 可用于定义回调方法。

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

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

 

三、委托的简单演化过程

  以下是 C# 声明和初始化委托的简单演化过程。

 1     class Program
 2     {
 3         //声明一个委托
 4         delegate void MyDel();
 5 
 6         static void Main(string[] args)
 7         {
 8             //以下是不同版本的声明和初始化委托的方式
 9 
10             // ≥ C# 1
11             MyDel del1 = new MyDel(Print);
12 
13             // ≥ C# 2
14             MyDel del2 = Print; //上面的简化版
15             MyDel del3 = delegate ()
16             {
17                 Console.WriteLine($"    {nameof(Print)}...");
18             };  //匿名方法
19 
20             // ≥ C# 3
21             MyDel del4 = () =>
22             {
23                 Console.WriteLine($"    {nameof(Print)}...");
24             };  //Lambda 表达式
25 
26             Console.Read();
27         }
28 
29         private static void Print()
30         {
31             Console.WriteLine($"    {nameof(Print)}...");
32         }
33     }

 

Use 委托

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

    //该委托可以封装 “,参数类型 string,返回类型 void” 的方法 
    public delegate void MyDel(string message);

 

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

 1     //该委托名为 MyDel,可以封装 “参数类型 string,返回值类型 void” 的方法 
 2     public delegate void MyDel(string message);
 3 
 4     class Program
 5     {
 6         static void Main(string[] args)
 7         {
 8             //实例化委托
 9             MyDel del = Print;
10             //调用委托
11             del("Hi");
12 
13             Console.Read();
14         }
15 
16         /// <summary>
17         /// 打印文本
18         /// </summary>
19         /// <remarks>这是一个可用于 MyDel 委托的方法</remarks>
20         /// <param name="message"></param>
21         private static void Print(string message)
22         {
23             Console.WriteLine(message);
24         }
25     }

异步回调,是在长进程完成时通知调用方的常用方法。当以这种方式使用委托时,使用委托的代码不需要知道要使用的实现方法。功能类似于封装接口提供的功能。       回调的另一个常见用途是定义自定义比较方法并将该委托传递到短方法。  它允许调用方的代码成为排序算法的一部分。  以下示例方法使用 Del 类型作为参数:  

 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             MyDel del = Print;
 6             CallbackMethod(100, 150, del);  //将委托传递到 CallbackMethod 方法
 7 
 8             Console.Read();
 9         }
10 
11         /// <summary>
12         /// 回调方法
13         /// </summary>
14         /// <param name="m"></param>
15         /// <param name="n"></param>
16         /// <param name="del"></param>
17         private static void CallbackMethod(int m, int n, MyDel del)
18         {
19             del((m + n).ToString());
20         }
21 
22         private static void Print(string message)
23         {
24             Console.WriteLine(message);
25         }
26     }

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

 1     //该委托可以封装 “名 MyDel,参数类型 string,返回值类型 void” 的方法 
 2     public delegate void MyDel(string message);
 3 
 4     class MyClass
 5     {
 6         public void Print1(string message)
 7         {
 8             Console.WriteLine($"{message} - {nameof(Print1)}");
 9         }
10 
11         public void Print2(string message)
12         {
13             Console.WriteLine($"{message} - {nameof(Print2)}");
14         }
15     }
16 
17     class Program
18     {
19         static void Main(string[] args)
20         {
21             var myClass = new MyClass();
22             MyDel del1 = myClass.Print1;
23             MyDel del2 = myClass.Print2;
24             MyDel del3 = Print;
25 
26             var del = del1 + del2;
27             del += del3;    //这里使用 +=
28             del("Hi!");
29 
30             Console.Read();
31         }
32 
33         private static void Print(string message)
34         {
35             Console.WriteLine($"{message} - {nameof(Print)}");
36         }
37     }

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

  此时,del 的调用列表中包含三个方法,分别为 Print1、Print2 和 Print。原有的三个委托(del1、del2 和 del3)保持不变。调用 allMethodsDelegate 时,将按顺序调用所有三个方法。如果委托使用引用参数,引用将按相反的顺序传递到所有这三个方法,并且一种方法进行的任何更改都将在另一种方法上见到。当方法引发未在方法内捕获到的异常时,该异常将传递到委托的调用方,并且不会调用调用列表中的后续方法。如果委托具有返回值和/或输出参数,它将返回上次调用方法的返回值和参数。若要删除调用列表中的方法,请使用减法运算符或减法赋值运算符(“-”或“-=”)。  例如:  

 1         static void Main(string[] args)
 2         {
 3             var myClass = new MyClass();
 4             MyDel del1 = myClass.Print1;
 5             MyDel del2 = myClass.Print2;
 6             MyDel del3 = Print;
 7 
 8             var del = del1 + del2;
 9             del += del3;    //使用 +=
10             del("Hi!");
11 
12             Console.WriteLine("======分割线======");
13 
14             del -= del2;    //使用 -=
15             del("Hi!");
16 
17             Console.Read();
18         }

1 static void Main(string[] args) 2 { 3 var myClass = new MyClass(); 4 MyDel del1 = myClass.Print1; 5 MyDel del2 = myClass.Print2; 6 MyDel del3 = Print; 7 8 var del = del1 + del2; 9 del += del3; //使用 += 10 //del("Hi!"); 11 12 var count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量 13 Console.WriteLine(count); 14 15 Console.WriteLine("======分割线======"); 16 17 del -= del2; //使用 -= 18 //del("Hi!"); 19 20 count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量 21 Console.WriteLine(count); 22 23 Console.Read(); 24 }

 


【参考】微软官方文档

] C# 知识回顾, C# 知识回顾 - 委托 delegate 【博主】反骨仔 【原文】 目录 What's 委托 委托的属性概述...

传送门

  《C# 知识回顾 - 序列化》

  《C# 知识回顾 - 表达式树 Expression Trees》

  《C# 知识回顾 - 特性 Attribute》、《剖析 AssemblyInfo.cs - 了解常用的特性 Attribute》

 


【参考】

【参考】微软官方文档

编辑:编程 本文来源:命名方法是通过参数的形式进行传递来进行委托

关键词: