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