委托的基础
委托和类一样,是一种用户自定义的类型。类表示的是数据的集合,而委托保存的是一个或多个方法。
1、申明委托类型
delegate void MyDel(int x)
上面代码申明了MyDel类型的委托,指明了这种类型的委托只会接受不返回值并且只有单个int参数的方法。
2、创建委托对象
委托是引用类型,因此有引用和对象。在委托类型申明之后,我们可以申明变量并创建类型的对象。
1 class del
2 {
3 public del()
4 {
5 MyDel delVar = new MyDel(b.GetB);//实例方法,创建委托并保存引用
6 }
7 delegate void MyDel(int x);//申明委托类型
8 B b = new B();
9 MyDel dVar = new MyDel(B.GetA);//实例方法,创建委托并保存引用
10 }
11 public class B
12 {
13 static public void GetA(int y)
14 {
15 Console.WriteLine("静态方法!输出值为:" + y);
16 }
17 public void GetB(int z)
18 { Console.WriteLine("成员方法!输出值为:" + z); }
19 }
上面代码创建了2个委托对象,一个具有实例方法,一个具有静态方法。b为类B的对象,它有一个叫做GetB实例方法,该方法接收一个int类型的参数,不返回值;B类有一个GetA静态方法,该方法具有与MyDel委托想匹配的返回类型和签名。
3、赋值托管
MyDel delVar;
delVar=b.GetB;//创建并给委托对象赋值……
delVar=B.GetA;//创建并给新的委托对象赋值
由于委托是引用类型,我们可以通过给它赋值来改变包含在委托变量中的引用,旧的委托对象会被垃圾回收器回收。
4、组合委托
MyDel delA=A.GetA;
MyDel delB=B.GetB;
MyDel delC=delA+delB;//组合调用列表
尽管术语组合委托让我们觉得好像操作委托被修改了,其实它们并没有被修改。事实上,委托是恒定的,委托被创建后不会再被改变。
5、为委托增加方法
MyDel delVar=A.GetA;//创建并初始化
delVar+=B.GetB;//增加方法
delVar+=C.GetC;//增加方法
委托是不可变的,所以委托的调用列表增加3个方法后的结果其实是变量指向一个全新的委托。
6、从委托移除方法
delVar-=C.GetC;//从委托移除方法
与为委托增加方法一样,其实是创建了一个新的委托,新委托是就委托的一个副本,只是没有了已移除方法的引用。
1.如果在调用列表中的方法有多个实例,-=运算符将从列表最后开始搜索,并且移除第一个与方法匹配的实例。
2.试图删除委托中不存在的方法没有结果。
3.视图调用空委托会抛出异常。
4.如果调用列表为空,则委托是null。
7、调用委托
1 delegate void printFunction();//定义一个没有返回值和参数的委托类型
2 class Program
3 {
4 static void Main(string[] args)
5 {
6 Test t = new Test();//创建测试类实例
7 printFunction pf;//创建一个空委托
8 pf = t.Print1;//实例化并初始化该委托
9 //给委托增加3个另外的方法
10 pf += Test.Print2;
11 pf += t.Print1;
12 pf += Test.Print2;
13 if (null != pf)//确认委托有方法
14 pf();//调用委托
15 else
16 Console.WriteLine("Delegage is empty");
17 }
18 }
19 class Test
20 {
21 public void Print1()
22 { Console.WriteLine("Print1--instance"); }
23 static public void Print2() { Console.WriteLine("Print2--static"); }
24 }
Test类定义了两个打印函数,一个是实例成员函数一个是静态成员函数,在调用委托之前,检测是否为null,程序打印结果如下图
8、调用带返回值的委托
如果委托有返回值并且在调用列表中有一个以上的方法,调用列表中最后一个方法返回的值就是委托调用返回的值,调用列表中所有其他方法的返回值都会被忽略。
1 delegate int Add();//定义一个有回值和无参数的委托类型
2 class Program
3 {
4 static void Main(string[] args)
5 {
6 Test t = new Test();//创建测试类实例
7 Add add = t.Add1;
8 add += t.Add2;
9 add += t.Add1;
10 Console.WriteLine(add());
11 }
12 }
13 class Test
14 {
15 int val = 5;
16 public int Add1()
17 { val += 2; return val; }
18 public int Add2()
19 {
20 val += 3; return val;
21 }
22 }
add();->Add1();返回值7被忽略->Add2();返回值10被忽略->Add1();使用,返回值为12
上面代码申明了返回int值的委托,然后调用委托打印出值,运行结果如图所示:
9、调用带引用参数的委托
如果委托有引用参数,参数值会根据调用列表的的一个或者多个方法的返回值而改变。
在调用委托列表下的方法时,参数的新值(不是初始值)会传给下一个方法。
1 delegate void MyDel(ref int x);//定义一个有回值和无参数的委托类型
2 class Program
3 {
4 public void Add1(ref int x) { x += 2; }
5 public void Add2(ref int x) { x += 3; }
6 static void Main(string[] args)
7 {
8 Program p = new Program();
9 MyDel mDel = p.Add1; mDel += p.Add2;
10 mDel += p.Add1;
11 int x = 5; mDel(ref x);
12 Console.WriteLine(x);
13 }
14 }
mDel();->Add1(X=5);引用x的初始值->Add2(X=7);引用x新的输入值->Add1(x=10);引用x的新的输入值
打印如图所示:
10、匿名方法
匿名方法:是子初始化委托时内联申明的方法。
1 delegate int MyDel(int x);//定义一个有回值和无参数的委托类型
2 class Program
3 {
4 static void Main(string[] args)
5 {
6 MyDel del = delegate(int x)
7 { return x += 20; };
8 Console.WriteLine(del(123));
9 Console.WriteLine(del(1234));
10 }
11 }
如果匿名方法申明的参数列表中包含了params参数,那么parmas关键字就会被匿名方法的参数列表忽略
delegate int MyDel(int x, params int[] y);//定义一个有回值和参数的委托类型
class Program
{
static void Main(string[] args)
{
MyDel del = delegate(int x, int[] y) { return x + y.Length; };
Console.WriteLine(del(1, 1, 2, 3, 4));
}
}
打印结果如下图示:
11.泛型委托
Func<int, int, int> FunAdd = (arg1, arg2) => arg1 + arg2;
Console.Write(FunAdd(5,6));
Action<int, int> ActionAdd= (arg1, arg2) => { Console.Write(arg1 + arg2); };
ActionAdd(5,6);
12、Lambda表达式
在匿名方法的语法中,delegate关键字是有点多余的,因为编译器已经知道我们将在方法赋值给委托,下面把匿名方法转换为lambda表达式
delegate int MyDel(int x);
class Program
{
static void Main(string[] args)
{
MyDel del = delegate(int x) { return x + 1; };//匿名函数
MyDel del1 = (int x) => { return x + 1; };//lambad表达式
MyDel del2 = (x) => { return x + 1; }; MyDel del3 = x => { return x + 1; }; MyDel del4 = x => x + 1; Console.WriteLine(del(12)); Console.WriteLine(del1(12)); Console.WriteLine(del2(12)); Console.WriteLine(del3(12)); Console.WriteLine(del4(12));
}
}