C#中的委托介绍
什么是委托?
之前写了事件的介绍:
这里也把委托相关知识也总结一下。
委托是c#中类型安全的,可以订阅一个或多个具有相同签名方法的函数指针
声明委托的方式:delegate 返回值类型 委托类型名(参数)
比如:
delegate void stringprocess(string s);
注意:这里的除了前面的delegate,剩下部分和声明一个函数一样,但是stringprocess不是函数名,而是委托类型名
声明的委托是一种类型,就像int、person一样,如果要用的话还要声明委托类型的变量,声明委托类型变量的方式:stringprocess f1;
将委托类型变量指向函数 stringprocess sp = new stringprocess(sayhello),这样就可以像调用普通函数一样把sp当成函数用了。委托可以看做是函数的指针。整数可以用整数变量指向它,对象可以用对象变量指向它,函数也可以用委托变量指向它。和直接调用函数的区别:用委托就可以指向任意的函数,哪怕是之前没定义的都可以,而不使用受限于那几种。
将委托类型变量指向函数还可以简化成stringprocess sp = sayhello,编译器帮我们进行了new。但是不能sp=printit(),因为这样就成了函数调用。
using system;
using system.collections.generic;
using system.linq;
using system.text;
using system.threading.tasks;
namespace wolfy.delegatetest
{
class program
{
delegate void stringprocess(string s);
static void main(string[] args)
{
string s = "hi,i am wolfy";
string s2 = "how do you do?";
//定义委托变量p
stringprocess p = new stringprocess(tolower);
//简化方式
stringprocess p2 = tolower;
p(s);
p2(s2);
console.read();
}
static void tolower(string s)
{
console.writeline(s.tolower());
}
}
}
结果:
stringprocess p2 = tolower;
这种简化方式,编译器到底为我们做了什么?可以通过反编译工具看一下:
可以见编译器帮咱们new了一个。
返回类型和签名指定了委托的接受方法的形式:也就是你的委托是什么样式的,指向的方法也要是什么样式的,比如这里返回类型为void 参数是string类型的。
匿名方法
使用delegate的时候很多时候没必要使用一个普通的方法,因为这个方法只有这个delegate会用,并且只用一次,这时候使用匿名方法最合适。
匿名方法就是没有名字的方法。3就是没有名字的int对象。3+5就是两个匿名int对象的相加,允许匿名对象,就允许匿名方法。
processworddelegate p = delegate(string s)
{
console.writeline(s);
};
知道c#中有匿名方法,看到这种写法知道是匿名函数即可
多播委托*
委托的增减方法
d+=sayhello,为委托增加一个方法,不要感觉奇怪,因为它就是d=d+ sayhello
d-=sayhello,将方法从委托中移除。
注意:在使用多播委托的时候,一般是让委托指向返回值为void的方法,不然会覆盖返回值,输出最后一个方法的返回值。
class program
{
delegate int intprocess(int a, int b);
static void main(string[] args)
{
intprocess p = add;
p += minus;
int result = p(1, 2);
console.write(result);
console.read();
}
static int add(int a, int b)
{
return a + b;
}
static int minus(int a, int b)
{
return a - b;
}
}
结果为:-1。
从结果可以看出是返回了最后一个方法minus。
委托的本质
其实就是一个类把方法包装了一下,委托都继承自system.multicastdelegate,而system.multicastdelegate又继承自system.delegate
多播委托就是有一个委托数组,依次调用