1、什么是委托
從數據結構來講,委托是和類一樣是一種用戶自定義類型。
委托是方法的抽象,它存儲的就是一系列具有相同簽名和返回回類型的方法的地址。調用委托的時候,委托包含的所有方法將被執行。
2、委托的定義
委托是類型,就好像類是類型一樣。與類一樣,委托類型必須在被用來創建變量以及類型對象之前聲明。
委托的聲明原型是
delegate <函數返回類型> <委托名> (<函數參數>)
例子:public delegate void MyDelegate(int number);//定義了一個委托MyDelegate,它可以注冊返回void類型且有一個int作為參數的函數
3、委托的實例化
3.1 使用new關鍵字
委托實例化的原型是
<委托類型> <實例化名>=new <委托類型>(<注冊函數>)
例子:MyDelegate _MyDelegate=new MyDelegate(CheckMod);//用函數CheckMod實例化上面的MyDelegate 委托為_MyDelegate
3.2 使用匿名方法
<委托類型> <實例化名>=delegate(<函數參數>){函數體};
3.3 使用Lambda表達式
1 class Program 2 { 3 //聲明委托 4 delegate int MyDelegate(int x, int y); 5 6 static void Main(string[] args) 7 { 8 //實例化委托 9 //1、使用new關鍵字 10 MyDelegate _myDelegate = new MyDelegate(GetSum); 11 12 //2、使用匿名方法 13 MyDelegate myDelegate = delegate(int x, int y) 14 { 15 return x + y; 16 }; 17 18 //3、使用Lambda表達式 19 MyDelegate myDelegateLambda = (int x, int y) => { return x + y; }; 20 } 21 22 static int GetSum(int x, int y) 23 { 24 return x + y; 25 } 26 }
4、泛型委托
委托也支持泛型的使用
泛型委托原型:
delegate <T1> <委托名><T1,T2,T3...> (T1 t1,T2 t2,T3 t3...)
例如:delegate T2 DelegateDemo<T1,T2>(T1 t);//定義有兩個泛型(T1,T2)的委托,T2作為委托函數返回類型,T1作為委托函數參數類型
static boo Check(int i)
{
if(i%2==0)
{
return true;
}
return false;
}
static void Main(string[] args)
{
DelegateDemo<int, bool> _delegate =Check;//將泛型委托委托<T1,T2>實例化為<int,bool>,即表示有一個int類型參數且返回類型是bool的函數.
Console.WriteLine(_delegate(9));//false
}
5、C#內置泛型委托
C#共有3種內置泛型委托
1 namespace DelegateDemo 2 { 3 class Program 4 { 5 //聲明委托 6 delegate int MyDelegate(int x, int y); 7 8 static void Main(string[] args) 9 { 10 //1、Action<T>只能委托必須是無返回值的方法 11 Action<string> _action = new Action<string>(SayHello); 12 _action("Hello World"); 13 14 //2、Fun<TResult>只是委托必須有返回值的方法 15 Func<int, bool> _func = new Func<int, bool>(Check); 16 _func(5); 17 18 //3、Predicate:此委托返回一個bool值,該委托通常引用一個"判斷條件函數"。 19 //需要指出的是,判斷條件一般為“外部的硬性條件”,比如“大於50”,而不是由數據自身指定,如“查找數組中最大的元素就不適合”。 20 Predicate<int> _predicate = new Predicate<int>(Check); 21 //使用Lambda表達式 22 Predicate<int> predicate = p => p % 2 == 0; 23 _predicate(26); 24 } 25 26 static void SayHello(string strMsg) 27 { 28 Console.WriteLine(strMsg); 29 } 30 31 //返回值為bool值 32 static bool Check(int i) 33 { 34 if (i % 2 == 0) 35 { 36 return true; 37 } 38 return false; 39 } 40 } 41 }
6、多播委托
實例化委托時必須將一個匹配函數注冊到委托上來實例化一個委托對象,但是一個實例化委托不僅可以注冊一個函數還可以注冊多個函數,注冊多個函數后,在執行委托的時候會根據注冊函數的注冊先后順序依次執行每一個注冊函數。
函數注冊委托的原型:
<委托類型> <實例化名>+=new <委托類型>(<注冊函數>)
例如:MyDelegate _myDelegate+=new MyDelegate(CheckMod);//將函數CheckMod注冊到委托實例_checkDelegate上
在.net 2.0開始可以直接將匹配的函數注冊到實例化委托:
<委托類型> <實例化名>+=<注冊函數>
例如:MyDelegate _myDelegate+=CheckMod;//將函數CheckMod注冊到委托實例_myDelegate上
注意:委托必須先實例化以后,才能使用+=注冊其他方法。如果對注冊了函數的委托實例從新使用=號賦值,相當於是重新實例化了委托,之前在上面注冊的函數和委托實例之間也不再產生任何關系。
有+=注冊函數到委托,也有-=解除注冊
例如:MyDelegate _myDelegate-=CheckMod;
如果在委托注冊了多個函數后,如果委托有返回值,那么調用委托時,返回的將是最后一個注冊函數的返回值。
1 namespace DelegateDemo 2 { 3 class Program 4 { 5 //聲明委托 6 delegate int MyDelegate(int x, int y); 7 8 static void Main(string[] args) 9 { 10 MyDelegate _myDelegate = new MyDelegate(fun1); 11 _myDelegate += fun2; 12 Console.WriteLine(_myDelegate(10,23)); 13 14 Console.ReadKey();//輸出10,返回最后一個注冊函數的返回值 15 } 16 17 static int fun1(int x, int y) 18 { 19 return x + y; 20 } 21 22 static int fun2(int x, int y) 23 { 24 return x; 25 } 26 } 27 }
