一、简介
委托是一种类型,由关键字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 }
其实Action与Func的用法差不多,差别只是一个有返回类型,一个没有返回类型,当然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类型