0x01定义一个委托,相当于定义一个可以存储方法的特殊变量类型
下面我们看具体的代码,通过代码更好理解
delegate void IntMethodInvoker(int x);
这行代码就是声明一个委托,其中delegate
是关键字,表示声明一个委托,void
是要存储的方法的返回类型,IntMethodInvoker
是声明的委托类型名字,结合最开始的那句话就是自定义的类书变量类型名,int x
则是要存储的方法的返回类型
这样一个委托可以存放哪些方法呢?
只要返回类型一致,参数列表一致的方法都能存储
在这个例子中,就是返回类型为void,有一个int形参的方法
void test(int t)
{
Console.WriteLine("Just a test" + t )
}
如这样的test方法就可以存储到IntMethodInvoker
委托的实例中
委托是一种特殊类型的对象,定义好委托之后就可以创建它的实例了然后存储方法
IntMethodInvoker DelegateTest = test;
这行代码即将test方法的地址存放到IntMethodInvoker
委托的实例DelegateTest
中,在以后只需要调用DelegateTest
就能执行test方法。
DelegateTest(32);
这行代码与test(32)
效果相同
委托可以添加多个方法引用,添加多个方法引用的委托叫多播委托
添加多个方法引用使用+=,若要删除添加的方法引用使用-=
0x02使用委托可以将方法当作参数传递
既然前面说了委托类似于定义一个可以存储方法的特殊变量类型,那么我们也可以将这个特殊类型作为方法的形参,将一个方法引用添加到委托实例中,然后将委托实例作为另一个接受委托类型参数的方法的实参,就达到了将方法传递给另一个方法的效果
class Program
{
static void Main(string[] args)
{
Test test = new Test();
Test.TestDelegate testDelegate = new Test.TestDelegate(test.Fun);
test.KK(testDelegate);
}
}
class Test
{
public delegate void TestDelegate();
public void Fun()
{
Console.WriteLine("这是一个简单的方法!");
}
public void KK(TestDelegate testDelegate)
{
Console.WriteLine("这是一个接收委托的方法");
testDelegate();
}
}
这是一个将委托作为参数,将一个方法传递给另一个方法的例子
在这个例子中,先声明了一个Test实例,然后声明了一个TestDelegate
委托实例testDelegate
testDelegate
委托包含了test.Fun()
的引用,这时并没有调用test.Fun()
,只是保存了引用
接下来执行test.KK(testDelegate)
,这时testDelegate
委托作为形参传递到了KK方法中
KK方法在调用testDelegate();
时,委托执行Fun方法
所以最后的执行结果是
这是一个接收委托的方法
这是一个简单的方法!
0x03Func<T>
和Action<T>
委托非常常用,所以.Net内置了两个委托类型,这样我们就不用自己去声明委托了
Action<T>
泛型委托,表示引用一个void返回类型的方法。这个委托存在不同的变体,可以接受需要0 ~ 16个参数的方法
Func<T>
泛型委托,与Action<T>
相似,但是表示引用一个有一个返回值类型的方法。Func<T>
同样可以接收0 ~ 16个参数的方法,默认最后一个是返回类型,即Fun<T>
最多有十七位参数
Func<string,string,int>
这个Func表示可以接收一个参数列表为两个string,返回一个int的方法。
0x03Lambda和匿名方法
匿名方法:
Func<string, string> anonDel = delegate (string param)
{
param += mid;
param += " and this was added to the string.";
return param;
};
匿名方法的作用:
委托是存储方法的引用,也就是说,要用委托,就必须要先有方法,但是实际上有很多时候委托所存储的方法引用只会在委托或者事件(事件基于委托)中使用,这时候我们单独定义一个方法很麻烦,这时候就可以使用匿名方法,上方就是一个匿名方法,我们看右边
delegate (string param)
{
param += mid;
param += " and this was added to the string.";
return param;
};
这个就是匿名方法,使用delegate
关键字,如果我们把delegate
换成一个普通的名字
KK (string param)
{
param += mid;
param += " and this was added to the string.";
return param;
};
这就是一个普通的方法(结尾的分号是Fun委托赋值语句的结尾),所以匿名方法就是一个普通的方法将方法名去掉换成delegate,以达到简化代码的目的
Func<string, string> anonDel =
delegate (string param)
{
param += mid;
param += " and this was added to the string.";
return param;
};
这样看是不是很好理解了
Lambda表达式:
在我的理解中,Lambda表达式其实是为了简化匿名方法
我们依然用上面的代码
//匿名方法
delegate (string param)
{
param += mid;
param += " and this was added to the string.";
return param;
}
//Lambda表达式
(string pram) =>
{
param += mid;
param += " and this was added to the string.";
return param;
}
这两个是等价版本, =>就是Lambda运算符,现在我们看它的两边是什么,左边是参数列表,右边是方法体。
这个Lambda其实有一点复杂了,通常我们遇到的Lambda会更简化
- 如果只有一个参数,只需要参数名就足够了,即这个Lambda中,可以将(string pram)改写为pram
- 如果方法体只有一行语句,那么在方法块中就不需要花括号和return语句
举个例子
Func<double,double> square = (double x) =>
{
return x*x;
};
这个Lambda表达式可以简写为
Func<double,double> square = x => x*x
Lambda表达式的闭包问题
这里就一句话,在Lambda表达式使用了外部的变量(不是Lambda表达式内的变量),外部的变量的值在调用时决定,而不是定义Lambda表达式时决定
int someVal = 5;
Func<int, int> ZZ = x => x + someVal;
someVal = 7;
Console.WriteLine(ZZ(3));
最后输出的时7,因为ZZ委托调用Lambda表达式时someVal已经等于7了