在 2.0 之前的 C# 版本中,声明委托的唯一方法是使用命名方法。C# 2.0 引入了匿名方法,而在 C# 3.0 及更高版本中,Lambda 表达式取代了匿名方法,作为编写内联代码的首选方式。
下面是C#各版委托写法:
一、在 C# 1.0 及更高版本中,可以按此处所示方式声明委托:
public delegate void Del<T>(T item);
public void Notify(int i) { }
Del<int> d1 = new Del<int>(Notify);
二、在 C# 2.0 及更高版本中,还可以使用以下简化的语法,通过匿名方法来声明和初始化
Del<int> d2 = Notify;
三、在 C# 3.0 及更高版本中,还可以使用 Lambda 表达式来声明和实例化委托。
delegate int del(int i);
del myDelegate = x => x * x;
int j = myDelegate(5); //j = 25
下面是一些规则和使用:
要将代码块传递为委托参数,创建匿名方法则是唯一的方法。这里是两个示例:
button1.Click += delegate(System.Object o, System.EventArgs e)
{ System.Windows.Forms.MessageBox.Show("Click!"); };
或下面
delegate void Del(int x);
Del d = delegate(int k) { /* 代码... */ };
通过使用匿名方法,由于您不必创建单独的方法,因此减少了实例化委托所需的编码系统开销。
例如,如果创建方法所需的系统开销是不必要的,则指定代码块(而不是委托)可能非常有用。启动新线程即是一个很好的示例。无需为委托创建更多方法,线程类即可创建一个线程并且包含该线程执行的代码。
void StartThread()
{
System.Threading.Thread t1 = new System.Threading.Thread
(delegate()
{
System.Console.Write("Hello, ");
System.Console.WriteLine("World!");
});
t1.Start();
}
匿名方法的参数的范围是“匿名方法块”。 Lambda也差不多有下面的规则
如果目标在块外部,那么,在匿名方法块内使用跳转语句(如 goto、break 或 continue)是错误的。如果目标在块内部,在匿名方法块外部使用跳转语句(如 goto、break 或 continue)也是错误的。
如果局部变量和参数的范围包含匿名方法声明,则该局部变量和参数称为该匿名方法的“外部”变量。
例如,下面代码段中的 n 即是一个外部变量:
C# 复制代码 int n = 0; Del d = delegate() { System.Console.WriteLine("Copy #:{0}", ++n); };
与局部变量不同,捕获变量的生命周期一直持续到引用该匿名方法的委托符合垃圾回收的条件为止。对 n 的引用是在创建该委托时捕获的。
规则:(Lambda也差不多有下面的规则)
- 匿名方法不能访问外部范围的 ref 或 out 参数。
- 在“匿名方法块”中不能访问任何不安全代码。
- 在 is 运算符的左侧不允许使用匿名方法。
- 匿名方法、 Lambda 语句无法用于创建表达式目录树。
所以匿名方法就像是为委托而生。Lambda为简化代码和匿名方法的替代者,以及在Linq查询等

