C# 委托
一、简介
委托是一种类型,由关键字delegate声明。确切的说,委托是一种可用于封装命名或者匿名方法的引用类型。 它类似于 C++ 中的函数指针,而且是类型安全和可靠的。
委托类型的声明与方法签名相似,有一个返回值和任意数目任意类型的参数。必须使用具有兼容返回类型和输入参数的方法或 lambda 表达式实例化委托。
委托允许将方法作为参数进行传递。
委托可用于定义回调方法。
委托可以链接在一起;例如,可以对一个事件调用多个方法。
方法不必与委托签名完全匹配。
二、Delegate
Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型。
public delegate int MethodDelegate(int x, int y); private static MethodDelegate method; static void Main(string[] args) { method = new MethodDelegate(Add); Console.WriteLine(method(10,20)); Console.ReadKey(); } private static int Add(int x, int y) { return x + y; }
三、Func<T>
在使用 Func<T, TResult> 委托时,不必显式定义一个封装只有一个参数的方法的委托。
以下示例简化了此代码,它所用的方法是实例化 Func<T, TResult> 委托,而不是显式定义一个新委托并将命名方法分配给该委托。
public class GenericFunc { public static void Main() { // 依旧是用命名方法实例化委托类型 Func<string, string> convertMethod = UppercaseString; string name = "Dakota"; // 依旧是通过委托实例调用该方法 Console.WriteLine(convertMethod(name)); } private static string UppercaseString(string inputString) { return inputString.ToUpper(); } }
下面的示例演示如何声明和使用 Func<T, TResult> 委托。
此示例声明一个 Func<T, TResult> 变量,并为其分配了一个将字符串中的字符转换为大写的 lambda 表达式。
随后将封装此方法的委托传递给Enumerable.Select 方法,以将字符串数组中的字符串更改为大写。
static class Func { static void Main(string[] args) { // 声明了一个Func委托类型的变量selector并用Lambda表达式进行实例化 // 这个Lambda表达式将用来获取一个字符串并将这个字符串转化为大写并返回 Func<string, string> selector = str => str.ToUpper(); // 创建一个字符串数组 string[] words = { "orange", "apple", "Article", "elephant" }; // 依次遍历这个字符串数组并调用委托实例selector进行处理 IEnumerable<String> aWords = words.Select(selector); // 输出结果到控制台 foreach (String word in aWords) Console.WriteLine(word); } } /* This code example produces the following output: ORANGE APPLE ARTICLE ELEPHANT */
四、Action<T>
Action 委托:没有传入参数,也没有返回类型,即Void。如:
void Main(string[] args) { Action say = SayHello; say(); } public static void SayHello( ) { Console.WriteLine("Say Hello"); }
Action<T> 委托:传入参数为T,没有返回类型。如:
void Main(string[] args) { Action<string> say = SayHello; say("Hello"); } public static void SayHello(string word ) { Console.WriteLine(word); }
其实Action与Func的用法差不多,差别只是一个有返回类型,一个没有返回类型,当然Action也可以接匿名方法和Lambda表达式。
匿名方法:
void Main(string[] args) { Action<string> say = delegate(string word) { Console.WriteLine(word); }; say("Hello Word"); }
Lambda表达式:
static void Main(string[] args) { Action<string> say = s => Console.WriteLine(s); say("Hello Word"); }
五、Predicate<T>
泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由 Array 和 List 类的几种方法使用,用于在集合中搜索元素。
void Main(string[] args) { Point[] points = { new Point(100, 200), new Point(150, 250), new Point(250, 375), new Point(275, 395), new Point(295, 450) }; Point first = Array.Find(points, ProductGT10); Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y); Console.ReadKey(); } private static bool ProductGT10(Point p) { if (p.X * p.Y > 100000) { return true; } else { return false; } }
使用带有 Array.Find 方法的 Predicate 委托搜索 Point 结构的数组。
如果 X 和 Y 字段的乘积大于 100,000,此委托表示的方法 ProductGT10 将返回 true。
Find 方法为数组的每个元素调用此委托,在符合测试条件的第一个点处停止。
六、总结
Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型
Func可以接受0个至16个传入参数,必须具有返回值
Action可以接受0个至16个传入参数,无返回值
Predicate只能接受一个传入参数,返回值为bool类型
参考文章:
http://www.cnblogs.com/xcsn/p/4520081.html
http://www.cnblogs.com/newbies/p/delegate.html
https://msdn.microsoft.com/zh-cn/library/bb534960.aspx
http://www.cnblogs.com/Gyoung/archive/2013/04/04/2997050.html
http://www.cnblogs.com/zyqgold/archive/2011/01/02/1924005.html
http://www.cnblogs.com/akwwl/p/3232679.html