C# 委托
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);
delegate string getString(string input); private static string result = ""; public static string getFirst(string str) { if (!String.IsNullOrEmpty(str)) { string target = str.ToCharArray()[0] + ""; result += target; return target; } return ""; } public static string getEnd(string str) { if (!String.IsNullOrEmpty(str)) { string target = str.ToCharArray()[str.Length - 1] + ""; result += target; return target; } return ""; } public static string getResult() { return result; } public static void run() { getString gs1 = new getString(getFirst); getString gs2 = new getString(getEnd); gs1("4114514"); Console.WriteLine(getResult()); gs2("114512"); Console.WriteLine(getResult()); }
委托的多播(Multicasting of a Delegate)
委托对象可使用 "+" 运算符进行合并。一个合并委托调用它所合并的两个委托。只有相同类型的委托可被合并。"-" 运算符可用于从合并的委托中移除组件委托。
使用委托的这个有用的特点,您可以创建一个委托被调用时要调用的方法的调用列表。这被称为委托的 多播(multicasting),也叫组播。
getString gs; getString gs1 = new getString(getFirst); getString gs2 = new getString(getEnd); gs = gs1 + gs2;//多播 gs("41145142"); Console.WriteLine(getResult());
delegate string getString(string input); public delegate void toDo(String msg); public static void writeLine(string msg) { Console.WriteLine(msg); } public static void writeAgain(string msg) { Console.WriteLine(msg); Console.WriteLine(msg); } public static void execute(toDo todo) { //实现委托 todo("Hello World."); } public static void run() { toDo td = new toDo(writeLine); toDo td1 = new toDo(writeAgain); execute(td); execute(td1); } }