在 C# 2.0 之前的版本中,创建委托的唯一方式是使用命名方法。 从 C# 2.0 开始引入了匿名方法,而在 C# 3.0 以及更高的版本中,Lambda 表达式取代了匿名方法,从而作为编写内联代码的首选方式。 不过,这里有关匿名方法的信息同样也适用于 Lambda 表达式。 需要注意的是,有一种情况,匿名方法它提供了 Lambda 表达式中所没有的功能。 就是,你可以使用匿名方法来忽略参数列表。 这意味着匿名方法可以转换为具有各种签名的委托。 这对于 Lambda 表达式来说几乎是不可能的。 有关 lambda 表达式的更多信息,我们下次再聊。
要将代码块传递为委托参数,创建匿名方法则是唯一的方法。
示例一:
// 创建一个点击事件 button1.Click += delegate(Object o, EventArgs e) { MessageBox.Show("Click!"); };
示例二:
// 创建一个委托. delegate void MyDel(int x); // 使用匿名方法实例化委托 MyDel d = delegate(int k) { /* ... */ };
通过使用匿名方法,由于不必创建单独的方法,从而减少了实例化委托所需的编码系统开销。
例如,如果创建方法所需的系统开销是不必要的,则指定代码块(而不是委托)可能非常有用。 这个启动新线程的示例即是一个很好的示例。 无需为委托创建更多方法,线程类即可创建一个线程并且包含该线程执行的代码。
void StartThread() { Thread t1 = new Thread (delegate() { Write("Hello, "); WriteLine("World!"); });
t1.Start(); }
匿名方法的参数的使用范围是“匿名方法块”。
如果目标在块外部,则在匿名方法块内使用可以跳转的语句(如 goto、break 或 continue)是错误的。 如果目标在块内部,在匿名方法块外部使用跳转的语句(如 goto
、break
或 continue
)也是错误的。
如果局部变量和参数的范围已经包含匿名方法声明,则该局部变量和参数将被称为该匿名方法的“外部”变量。 比如,这个代码段中的 n
就是一个外部变量:
int n = 0; MyDel d = delegate() { WriteLine("#:{0}", ++n); };
外部变量的引用 n,将
被认为是在创建委托时进行捕获,它与本地变量不同,所被捕获的变量的在生存期内,会存活到引用该匿名方法委托被垃圾回收。
匿名方法不能访问外部范围的 ref 或 out 参数。
在“匿名方法块”中不能访问任何不安全代码。
在 is 运算符的左侧不允许使用匿名方法。
下面的示例演示实例化委托的方法,这两种方法都会在调用委托时显示一条消息:
使委托与匿名方法关联。
使委托与命名方法 (DoWork
) 关联。
// 声明委托 delegate void Printer(string s); class MyClass { static void Main() { // 使用匿名方法实例化委托 Printer p = delegate(string j) { WriteLine(j); }; // 返回匿名委托所调用的结果 p("使用匿名方法调用委托。"); // 使用"Do"的命名的方法的委托实例化。 p = new Printer(TestClass.Do); // 使用旧的风格调用委托 p("使用命名方法调用委托。"); } // 后续用于委托调用 static void Do(string msg) { WriteLine(msg); } }