Latest web development tutorials

C# 委託(Delegate)

C# 中的委託(Delegate)類似於C 或C++ 中函數的指針。委託(Delegate)是存有對某個方法的引用的一種引用類型變量。引用可在運行時被改變。

委託(Delegate)特別用於實現事件和回調方法。 所有的委託(Delegate)都派生自System.Delegate類。

聲明委託(Delegate)

委託聲明決定了可由該委託引用的方法。 委託可指向一個與其具有相同標籤的方法。

例如,假設有一個委託:

public delegate int MyDelegate (string s);

上面的委託可被用於引用任何一個帶有一個單一的string參數的方法,並返回一個int類型變量。

聲明委託的語法如下:

delegate <return type> <delegate-name> <parameter list>

實例化委託(Delegate)

一旦聲明了委託類型,委託對象必須使用new關鍵字來創建,且與一個特定的方法有關。 當創建委託時,傳遞到new語句的參數就像方法調用一樣書寫,但是不帶有參數。 例如:

public delegate void printString(string s);
...
printString ps1 = new printString(WriteToScreen);
printString ps2 = new printString(WriteToFile);

下面的實例演示了委託的聲明、實例化和使用,該委託可用於引用帶有一個整型參數的方法,並返回一個整型值。

using System;

delegate int NumberChanger(int n);
namespace DelegateAppl
{
   class TestDelegate
   {
      static int num = 10;
      public static int AddNum(int p)
      {
         num += p;
         return num;
      }

      public static int MultNum(int q)
      {
         num *= q;
         return num;
      }
      public static int getNum()
      {
         return num;
      }

      static void Main(string[] args)
      {
         // 創建委託實例NumberChanger nc1 = new NumberChanger(AddNum);
         NumberChanger nc2 = new NumberChanger(MultNum);
         // 使用委託對象調用方法nc1(25);
         Console.WriteLine("Value of Num: {0}", getNum());
         nc2(5);
         Console.WriteLine("Value of Num: {0}", getNum());
         Console.ReadKey();
      }
   }
}

當上面的代碼被編譯和執行時,它會產生下列結果:

Value of Num: 35
Value of Num: 175

委託的多播(Multicasting of a Delegate)

委託對象可使用"+" 運算符進行合併。 一個合併委託調用它所合併的兩個委託。 只有相同類型的委託可被合併。 "-" 運算符可用於從合併的委託中移除組件委託。

使用委託的這個有用的特點,您可以創建一個委託被調用時要調用的方法的調用列表。 這被稱為委託的多播(multicasting) ,也叫組播。 下面的程序演示了委託的多播:

using System;

delegate int NumberChanger(int n);
namespace DelegateAppl
{
   class TestDelegate
   {
      static int num = 10;
      public static int AddNum(int p)
      {
         num += p;
         return num;
      }

      public static int MultNum(int q)
      {
         num *= q;
         return num;
      }
      public static int getNum()
      {
         return num;
      }

      static void Main(string[] args)
      {
         // 創建委託實例NumberChanger nc;
         NumberChanger nc1 = new NumberChanger(AddNum);
         NumberChanger nc2 = new NumberChanger(MultNum);
         nc = nc1;
         nc += nc2;
         // 調用多播nc(5);
         Console.WriteLine("Value of Num: {0}", getNum());
         Console.ReadKey();
      }
   }
}

當上面的代碼被編譯和執行時,它會產生下列結果:

Value of Num: 75

委託(Delegate)的用途

下面的實例演示了委託的用法。 委託printString可用於引用帶有一個字符串作為輸入的方法,並不返回任何東西。

我們使用這個委託來調用兩個方法,第一個把字符串打印到控制台,第二個把字符串打印到文件:

using System;
using System.IO;

namespace DelegateAppl
{
   class PrintString
   {
      static FileStream fs;
      static StreamWriter sw;
      // 委託聲明public delegate void printString(string s);

      // 該方法打印到控制台public static void WriteToScreen(string str)
      {
         Console.WriteLine("The String is: {0}", str);
      }
      // 該方法打印到文件public static void WriteToFile(string s)
      {
         fs = new FileStream("c:\\message.txt",
         FileMode.Append, FileAccess.Write);
         sw = new StreamWriter(fs);
         sw.WriteLine(s);
         sw.Flush();
         sw.Close();
         fs.Close();
      }
      // 該方法把委託作為參數,並使用它調用方法public static void sendString(printString ps)
      {
         ps("Hello World");
      }
      static void Main(string[] args)
      {
         printString ps1 = new printString(WriteToScreen);
         printString ps2 = new printString(WriteToFile);
         sendString(ps1);
         sendString(ps2);
         Console.ReadKey();
      }
   }
}

當上面的代碼被編譯和執行時,它會產生下列結果:

The String is: Hello World