C#中有很多易混淆的关键词,例如delegate,Func, Action和 Predicate。Func, Action和 Predicate本质上都是delegate,下面看一下delegate概念。
delegate本质上就是一个指向函数的指针,可以指向不同的函数,只要函数的签名和代理一致即可。
其实Func, Action, Predicate等都是delegate,只是特殊的delegate而已。delegate的巧妙应用,可以大大简化代码和提高灵活性。下面有一段Javascript代码,JS中经常使用数组的each方法来遍历数组并对其进行处理,如下所示:
1 var arr = [ "one", "two", "three", "four"]; 2 $.each(arr, function(){ 3 alert(this); 4 }); 5 //上面这个each输出的结果分别为:one,two,three,four
那么在C#中如何通过delegate来定义一个数组each方法呢,可以通过传入方法来实现灵活的逻辑处理,静态ListEx类下有一个静态的Each方法,定义如下:
1 public static T[] Each<T>(T[] source, Func<T, T> function) 2 { 3 4 T[] ret =new T[source.Length]; 5 int i = 0; 6 foreach (T item in source) 7 { 8 ret[i]=function(item); 9 i++; 10 } 11 return ret; 12 }
那么我们可以定义一个字符串数组,并定义一个delegate作为函数参数进行传入,调用ListEx.Each方法:
1 var arr =new string[]{ "one", "two", "three", "four"}; 2 var newArr= ListEx.Each<string>(arr,delegate(string x){ 3 x=x+"_do"; 4 return x; 5 });
当然可以用表达式进行简化:
1 var newArr2 = ListEx.Each<string>(newArr, (string x) => x = x + "_do");
我们也可以定义一个Where方法来过滤数组:
1 public static IList<T> Find<T>(IList<T> source, Predicate<T> predicate) 2 { 3 List<T> ret = new List<T>(); 4 foreach (T item in source) 5 { 6 if (predicate(item)) 7 { 8 ret.Add(item); 9 } 10 } 11 return ret; 12 } 13 public static T[] Where<T>(T[] source, Predicate<T> predicate) 14 { 15 IList<T> list=source.ToList<T>(); 16 IList<T> retList= Find<T>(list, predicate); 17 return retList.ToArray<T>(); 18 }
调用如下:
1 var newArr3 = ListEx.Where<string>(arr, x => x == "two");
Func是必须指定返回值的代理;
Action为返回值为void的代理;
Predicate为返回值为bool的代理;