委託可以這樣來理解:它是一種資料類型,和參考型別類似,不過和一般的類相比,委託的執行個體不是在堆中的資料,而是一個方法。委託類似於參考型別,和C++中的函數指標很相似,但是不同之處就在於,它不是存在於堆中的。正是有了委託我們才得以將函數打包成一個變數進行傳遞。這就是說,我們不止能夠將資料作為參數傳遞,還可以將一個函數作為參數進行傳遞,這樣就可以在需要的時候方便的進行控制的反轉(Ioc,控制反轉)。
代碼
下面寫一個簡單的執行個體進行理解:兩個數求其中的較大值、較小值。
/// <summary> /// 兩個數求較大值 /// </summary> /// <param name="x"></param> /// <param name="y"></param> /// <returns></returns> int Max(int x, int y) { return x > y ? x : y; } /// <summary> /// 兩個數求較小值 /// </summary> /// <param name="x"></param> /// <param name="y"></param> /// <returns></returns> int Min(int x, int y) { return x < y ? x : y; } ///建立與上面求較大值和較小值具有相同簽名的一個委託 delegate int MyDelegate(int x, int y); MyDelegate md = null;
上面代碼先聲明了兩個函數Max和Min,然後聲明具有與上述兩函數相同簽名的委託 delegate int MyDelegate(int x, int y);
看上述輸入兩個數字,並選擇Max或者Min,單擊按鈕即可獲的結果。
private void RbMax_CheckedChanged(object sender, EventArgs e) { if (RbMax.Checked == true) { md = new MyDelegate(Max); } } private void RbMin_CheckedChanged(object sender, EventArgs e) { if (RbMin.Checked == true) { md = new MyDelegate(Min); } }
兩個RadioButton按鈕的事件可以看出。如果選擇了RbMax,則讓MyDelegate類型的執行個體“md”指向Max方法,如果使用者選擇了求最小值的RbMin,則讓MyDelegate類型的執行個體“md”指向Min方法。這樣作的目的,就是要把選擇的過程交給使用者。
private void BtnReturn_Click(object sender, EventArgs e) { if (this.md == null) { MessageBox.Show("委託md沒有指向任何方法!"); return; } int FirstValue = int.Parse(this.TbFirst.Text); int SecondValue = int.Parse(this.TbSecond.Text); MessageBox.Show(md(FirstValue, SecondValue).ToString()); }
輸入完兩個數字並選擇方式之後的單擊按鈕事件。
從上面的代碼中,可以發現,在使用委託之前,先要判斷其值是否為空白,如果不為空白,則可以進行調用,同時,使用者可以看到,在調用md時,我們並沒有關心md到底指向了哪一個方法,總之,md不為空白的時候,就一定會指向Max和Min當中的一個。
運行效果如下:
執行個體結束,總結以下幾點:
Ø 在C#中,所有的委託都是從System.MulticastDelegate類派生的。
Ø 委託隱含具有sealed屬性,即不能用來派生新的類型。
Ø 委託最大的作用就是為類的事件綁定事件處理常式。
Ø 在通過委託調用函數前,必須先檢查委託是否為空白(null),若非空,才能調用函數。
Ø 在委託執行個體中可以封裝靜態方法也可以封裝執行個體方法。
Ø 在建立委託執行個體時,需要傳遞將要映射的方法或其他委託執行個體以指明委託將要封裝的函數原型(.NET中稱為方法簽名:signature)。注意,如果映射的是靜態方法,傳遞的參數應該是類名.方法名,如果映射的是執行個體方法,傳遞的參數應該是執行個體名.方法名。
Ø 只有當兩個委託執行個體所映射的方法以及該方法所屬的對象都相同時,才認為它們是想等的(從函數地址考慮)。