Csharp 简明教程

C

C# 委托类似于 C 或 C++ 中的函数指针。 delegate 是保存对方法引用的引用类型变量,可在运行时更改该引用。

委托特别用于实现事件和回调方法。所有委托都隐式派生 System.Delegate 类。

Declaring Delegates

委托声明确定了委托可以引用的方法。委托可以引用与委托具有相同签名的某个方法。

比如,考虑一个委托 −

public delegate int MyDelegate (string s);

前面的委托可用于引用具有单个字符串参数并返回 int 类型变量的任何方法。

委托声明的语法为 −

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

Instantiating Delegates

委托类型声明后,必须使用 new * keyword and be associated with a particular method. When creating a delegate, the argument passed to the *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) {
         //create delegate instances
         NumberChanger nc1 = new NumberChanger(AddNum);
         NumberChanger nc2 = new NumberChanger(MultNum);

         //calling the methods using the delegate objects
         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) {
         //create delegate instances
         NumberChanger nc;
         NumberChanger nc1 = new NumberChanger(AddNum);
         NumberChanger nc2 = new NumberChanger(MultNum);

         nc = nc1;
         nc += nc2;

         //calling multicast
         nc(5);
         Console.WriteLine("Value of Num: {0}", getNum());
         Console.ReadKey();
      }
   }
}

编译并执行上述代码后,将产生以下结果 −

Value of Num: 75

Using Delegates

以下示例演示了委托的使用。委托 printString 可用于引用获取字符串作为输入且无返回值的方法。

我们使用此委托调用两个方法,第一个将字符串打印到控制台,第二个将其打印到文件 −

using System;
using System.IO;

namespace DelegateAppl {

   class PrintString {
      static FileStream fs;
      static StreamWriter sw;

      // delegate declaration
      public delegate void printString(string s);

      // this method prints to the console
      public static void WriteToScreen(string str) {
         Console.WriteLine("The String is: {0}", str);
      }

      //this method prints to a file
      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();
      }

      // this method takes the delegate as parameter and uses it to
      // call the methods as required
      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