1. 程式人生 > 程式設計 >c# 委託的常見用法

c# 委託的常見用法

  此篇文章是我一個小白對委託的理解和總結,請高手多多評判指教。

  委託就是一種後期繫結機制,說的直白點就是在呼叫的時候才去傳遞業務邏輯的一種演算法。

委託的建立語法:

public delegate int Comparison<in T>(T left,T right);//官方給出的定義泛型委託的demo

  語法看似像宣告一個變數或方法的簽名,但實現上是在宣告一個型別。編譯器會生成一個派生自System.MulticastDelegate的類(而System.MulticastDelegate派生自System.Delegate),型別名與委託的名字相同,其中包含Invoke 、BeginInvoke和EndInvoke等方法。編譯器還為這個新型別生成新增和刪除處理業務,以便該類的客戶端可以在例項的呼叫列表中新增和刪除方法。

  委託可以被定義在類的內部、名稱空間下(與類同級)和全域性名稱空間下(不推薦)。

//在全域性定義
public delegate int Comparison<in T>(T left,T right);
namespace Test
{
  //在指定名稱空間下定義
  public delegate int Comparison2<in T>(T left,T right);
  public class Student
  {
    //在類內部定義
    public delegate int Comparison3<in T>(T left,T right);
  }
}

委託的定義賦值

  將委託當成類使用(委託本身就是一個類)。

//定義委託
public delegate int Comparison<in T>(T left,T right);
public class Test
{
  //定義
  private Comparison<int> comparator;
  public void Show()
  {
    //賦值 請注意,使用的是方法名稱,不帶括號,將方法附加給委託作為委託的呼叫方法。
    this.comparator = Compare;
    //呼叫 
    this.comparator(1,2);     //呼叫方式二    this.comparator.Invoke(1,2);
  }
  private int Compare(int left,int right) => left.CompareTo(right);
}

當用作委託的目標方法是“小方法”的情況下,通常使用lambda表示式語法來執行賦值:

Comparison<string> comparer = (left,right) => left.Length.CompareTo(right.Length);

多播委託

  通常只是將單個目標方法附加到委託。但是,委託物件確實支援將多個目標方法附加到一個委託物件的呼叫列表,稱為多播委託。多播委託意味著通過委託呼叫時可以呼叫多個方法,所以可以為委託附加多個方法。

private int Cal(int num) {return num * num;}
private void button1_Click(object sender,EventArgs e)
{
   Func<int,int> action = null;
   //使用+= 或 -= 來增加或移除委託例項
   action += a => { Console.WriteLine($"第{1}次執行" + a); return a + 1; };
   action += Cal;
   action += a => { Console.WriteLine($"第{3}次執行" + a); return a + 3; };
   action -= Cal;
   //如果有返回值的話,返回的是最後一次執行的結果
   int a = action(5);
   Console.WriteLine(a);
}

常用泛型委託

無返回值

public delegate void Action();
public delegate void Action<in T>(T arg);
public delegate void Action<in T1,in T2>(T1 arg1,T2 arg2);
// Other variations removed for brevity.

有返回值

public delegate TResult Func<out TResult>();
public delegate TResult Func<in T1,out TResult>(T1 arg);
public delegate TResult Func<in T1,in T2,out TResult>(T1 arg1,T2 arg2);
// Other variations removed for brevity

返回bool型別

public delegate bool Predicate<in T>(T obj);

注意:在.net core平臺中,不支援委託目標方法的非同步呼叫(即不支援BeginInvoke方法):

 Action action = () => Console.WriteLine("委託執行了");
 AsyncCallback asyncCallback = a => Console.WriteLine("回撥執行了" + a);
 //System.PlatformNotSupportedException:“Operation is not supported on this platform.”
 IAsyncResult result = action.BeginInvoke(asyncCallback,"asdf"); //委託啟動非同步呼叫
 action.EndInvoke(result);

Event 事件

事件是帶event關鍵字的委託的例項,event可以限制被外部呼叫(invoke)和直接賦值(=)。委託是一個型別,而事件是委託型別的一個例項。

  宣告一個事件很簡單,只需在宣告一個委託物件時加上event關鍵字就行。

/// <summary>
/// 定義一個委託
/// </summary>
/// <param name="name"></param>
public delegate void ShowInfo(string name);
public class Study
{
  /// <summary>
  /// 宣告一個事件
  /// </summary>
  public event ShowInfo ShowInfo;
  public void Show()
  {
    ShowInfo += Study_ShowInfo;
    //只能在“publisher”類中呼叫
    ShowInfo("asdf");
  }
  private void Study_ShowInfo(string name)
  {
    throw new NotImplementedException();
  }
}

可以在用事件的地方用委託來替代,但事件有一系列規則和約束用以保證程式的安全可控,事件只有 += 和 -= 操作,這樣訂閱者只能有訂閱或取消訂閱操作,沒有許可權執行其它操作。如果是委託,那麼訂閱者就可以使用 = 來對委託物件重新賦值(其它訂閱者全部被取消訂閱),甚至將其設定為null,甚至訂閱者還可以直接呼叫委託,這些都是很危險的操作,廣播者就失去了獨享控制權。

事件保證了程式的安全性和健壯性。

以上就是c# 委託的常見用法的詳細內容,更多關於c# 委託的資料請關注我們其它相關文章!