首页 > 技术文章 > C# 委托简介

lgx5 2018-09-15 08:38 原文

一、简介

  委托是一种类型,由关键字delegate声明。确切的说,委托是一种可用于封装命名或者匿名方法的引用类型。  它类似于 C++ 中的函数指针,而且是类型安全和可靠的。

  委托类型的声明与方法签名相似,有一个返回值和任意数目任意类型的参数。必须使用具有兼容返回类型和输入参数的方法或 lambda 表达式实例化委托。

  委托允许将方法作为参数进行传递。
  委托可用于定义回调方法。
  委托可以链接在一起;例如,可以对一个事件调用多个方法。
  方法不必与委托签名完全匹配。

二、Delegate

   Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型。

 1 public delegate int MethodDelegate(int x, int y);
 2 private static MethodDelegate method;
 3 static void Main(string[] args)
 4 {
 5      method = new MethodDelegate(Add);
 6      Console.WriteLine(method(10,20));
 7      Console.ReadKey();
 8 }
 9 
10 private static int Add(int x, int y)
11 {
12      return x + y;
13 }

三、Func<T>

 在使用 Func<T, TResult> 委托时,不必显式定义一个封装只有一个参数的方法的委托。

 以下示例简化了此代码,它所用的方法是实例化 Func<T, TResult> 委托,而不是显式定义一个新委托并将命名方法分配给该委托。

 1 public class GenericFunc
 2 {
 3    public static void Main()
 4    {
 5       // 依旧是用命名方法实例化委托类型
 6       Func<string, string> convertMethod = UppercaseString;
 7       string name = "Dakota";
 8       // 依旧是通过委托实例调用该方法
 9       Console.WriteLine(convertMethod(name));
10    }
11 
12    private static string UppercaseString(string inputString)
13    {
14       return inputString.ToUpper();
15    }
16 }

下面的示例演示如何声明和使用 Func<T, TResult> 委托。  

此示例声明一个 Func<T, TResult> 变量,并为其分配了一个将字符串中的字符转换为大写的 lambda 表达式。  

随后将封装此方法的委托传递给Enumerable.Select 方法,以将字符串数组中的字符串更改为大写。

 1 static class Func
 2 {
 3    static void Main(string[] args)
 4    {
 5       // 声明了一个Func委托类型的变量selector并用Lambda表达式进行实例化 
 6       // 这个Lambda表达式将用来获取一个字符串并将这个字符串转化为大写并返回
 7       Func<string, string> selector = str => str.ToUpper();
 8 
 9       // 创建一个字符串数组
10       string[] words = { "orange", "apple", "Article", "elephant" };
11       // 依次遍历这个字符串数组并调用委托实例selector进行处理
12       IEnumerable<String> aWords = words.Select(selector);
13 
14       // 输出结果到控制台
15       foreach (String word in aWords)
16          Console.WriteLine(word);
17    }
18 }      

四、Action<T>

Action 委托:没有传入参数,也没有返回类型,即Void。如:

1 void Main(string[] args)
2  {
3     Action say = SayHello;
4     say();
5 }
6 public static void SayHello( )
7 {
8     Console.WriteLine("Say Hello");
9 }

Action<T> 委托:传入参数为T,没有返回类型。如:

1 void Main(string[] args)
2 {
3    Action<string> say = SayHello;
4    say("Hello");
5 }
6  public static void SayHello(string word )
7 {
8    Console.WriteLine(word);
9 }

其实ActionFunc的用法差不多,差别只是一个有返回类型,一个没有返回类型,当然Action也可以接匿名方法和Lambda表达式。

匿名方法:

1 void Main(string[] args)
2 {
3      Action<string> say = delegate(string word)
4      {
5         Console.WriteLine(word);
6      };
7      say("Hello Word");
8 }

Lambda表达式:

1  static void Main(string[] args)
2 {
3    Action<string> say = s => Console.WriteLine(s);
4    say("Hello Word");
5 }

五、Predicate<T>

 泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由 Array 和 List 类的几种方法使用,用于在集合中搜索元素。

 1 void Main(string[] args)
 2 {
 3      Point[] points = { new Point(100, 200), 
 4      new Point(150, 250), new Point(250, 375), 
 5      new Point(275, 395), new Point(295, 450) };
 6      Point first = Array.Find(points, ProductGT10);
 7      Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y);
 8      Console.ReadKey();
 9 }
10 private static bool ProductGT10(Point p)
11 {
12     if (p.X * p.Y > 100000)
13     {
14         return true;
15      }
16      else
17      {
18         return false;
19      }
20 }

使用带有 Array.Find 方法的 Predicate 委托搜索 Point 结构的数组。

如果 X 和 Y 字段的乘积大于 100,000,此委托表示的方法 ProductGT10 将返回 true。

Find 方法为数组的每个元素调用此委托,在符合测试条件的第一个点处停止。

六、总结

    Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型

  Func可以接受0个至16个传入参数,必须具有返回值

  Action可以接受0个至16个传入参数,无返回值

  Predicate只能接受一个传入参数,返回值为bool类型

推荐阅读