C# 2.0 Specification(匿名方法)(二)

21.7委托实例相等性
如下规则适用由匿名方法委托实例的相等运算符(§7.9.8)和object.Equals方法产生的结果。

l 当委托实例是由具有相同被捕获外部变量集合的语义相同的匿名方法表达式计算而产生时,可以说(但不是必须)它们相等。

l 当委托实例由具有语义不同的匿名方法表达式,或具有不同的被捕获外部变量集合时,它们决不相等。

21.8明确赋值
匿名方法参数的明确赋值状态与命名方法是相同的。也就是,引用参数和值参数被明确的赋初值,而输出参数不用赋初值。并且,输出参数在匿名方法正常返回之前必须被明确赋值(§5.1.6)。

当控制转换到匿名方法表达式的程序块时,对外部变量v的明确赋值状态,与在匿名方法表达式之前的v的明确赋值状态是相同的。也就是,外部变量的明确赋值将从匿名方法表达式上下文被继承。在匿名方法程序块内,明确赋值将和在普通程序块内一样而得到演绎(§5.3.3)。

在匿名方法表达式之后的变量v的明确赋值状态与在匿名方法表达式之前它的明确赋值状态相同。

例如

delegate bool Filter(int i);

void F() {
int max;

// 错误,max没有明确赋值
Filter f = delegate(int n) { return n < max; }

max = 5;
DoWork(f);
}

将产生一个编译时错误,因为max没有在匿名方法声明的地方明确赋值。示例

delegate void D();

void F() {
int n;
D d = delegate { n = 1; };

d();

//错误,n没有明确赋值
Console.WriteLine(n);
}

也将产生一个编译时错误,因为匿名方法内n的赋值,对于该匿名方法外部n的明确赋值状态没有效果。

21.9方法组转换
与在§21.3中描述的隐式匿名方法转换相似,也存在从方法组(§7.1)到兼容的委托类型的隐式转换。

对于给定的方法组E和委托类型D,如果允许new D(E)形式的委托创建表达式(§7.5.10.3 和 §20.9.6),那么就存在从E到D的隐式转换,并且转换的结果恰好等价于new D(E)。

在以下示例中

using System;
using System.Windows.Forms;

class AlertDialog
{
Label message = new Label();
Button okButton = new Button();
Button cancelButton = new Button();`

public AlertDialog() {
okButton.Click += new EventHandler(OkClick);
cancelButton.Click += new EventHandler(CancelClick);
...
}

void OkClick(object sender, EventArgs e) {
...
}

void CancelClick(object sender, EventArgs e) {
...
}
}

构造函数用new创建了两个委托实例。隐式方法组转换允许将之简化为

public AlertDialog() {
okButton.Click += OkClick;
cancelButton.Click += CancelClick;
...
}

对于所有其他隐式和显式的转换,转换运算符可以用于显式地执行一个特定的转换。为此,示例

object obj = new EventHandler(myDialog.OkClick);

可被代替写成如下的样子。

object obj = (EventHandler)myDialog.OkClick;

方法组合匿名方法表达式可以影响重载决策(overload resolution),但它们并不参与类型推断。请参见§20.6.4获取更详细的信息。

21.10实现例子
本节以标准C#的构件形式描述匿名方法的可能实现。在这里描述的实现基于Microsoft C#编译器所采用的相同原则,但它决不是强制性的或唯一可能的实现。

本节的后面部分给出了几个示例代码,它包含了具有不同特性的匿名方法。对于每个例子,我们将提供使用唯一标准C#构件的代码的对应转换。在这些例子中,标识符D假定表示如下委托类型。

public delegate void D();

匿名方法的最简形式就是没有捕获外部变量的那个。

class Test
{
static void F() {
D d = delegate { Console.WriteLine("test"); };
}
}

这段代码可被转换到一个引用编译器生成的静态方法的委托实例,而匿名方法的代码将会放入到该静态方法中。、

class Test
{
static void F() {
D d = new D(__Method1);
}

static void __Method1() {
Console.WriteLine("test");
}
}

在下面的示例中,匿名方法引用this的实例成员。

class Test
{
int x;

void F() {
D d = delegate { Console.WriteLine(x); };
}
}

this可以被转换到由编译器生成的包含匿名方法代码的实例方法。

class Test
{
int x;

void F() {
D d = new D(__Method1);
}

void __Method1() {
Console.WriteLine(x);
}
}

在这个例子中,匿名方法捕获了一个局部变量。

class Test
{
void F() {
int y = 123;
D d = delegate { Console.WriteLine(y); };
}
}

该局部变量的生存期现在至少必须延长到匿名方法委托的生存期为止。这可以通过将局部变量“提升(lifting)”为编译器生成的(compiler-generated)类的字段来完成。局部变量的实例化对应于创建一个编译器生成的类的实例,而访问局部变量将对应于访问编译器生成的类实例的一个字段。并且,匿名方法将成为编译器生成类的实例方法。

class Test
{
void F() {
__locals1 = new __Locals1();
__locals1.y = 123;
D d = new D(__locals1.__Method1);
}

class __Locals1
{
public int y;

public void __Method1() {
Console.WriteLine(y);
}
}
}

最后,如下匿名方法将捕获this,以及具有不同生存期的两个局部变量。

class Test
{
int x;

void F() {
int y = 123;
for (int i = 0; i < 10; i++) {
int z = i * 2;
D d = delegate { Console.WriteLine(x + y + z); };
}
}
}

在这里,编译器将为每个语句块生成类,在这些语句块中局部变量将被捕获,而在不同块中的局部变量将会有独立的生存期。

__Locals2的实例,编译器为内部语句块生成的类,包含局部变量z和引用__Locals1实例的字段。__Locals1的实例,编译器为外部语句块生成的类,包含局部变量y和引用封闭函数成员的this的字段。通过这些数据结构,你可以通过__Locals2的一个实例到达所有被捕获的局部变量,并且匿名方法的代码可以作为那个类的实例方法而实现。

class Test
{
void F() {
__locals1 = new __Locals1();
__locals1.__this = this;
__locals1.y = 123;
for (int i = 0; i < 10; i++) {
__locals2 = new __Locals2();
__locals2.__locals1 = __locals1;
__locals2.z = i * 2;
D d = new D(__locals2.__Method1);
}
}

class __Locals1
{
public Test __this;
public int y;
}

class __Locals2
{
public __Locals1 __locals1;
public int z;

public void __Method1() {
Console.WriteLine(__locals1.__this.x + __locals1.y + z);
}
}
}

(匿名方法完)

时间: 2024-10-28 04:59:15

C# 2.0 Specification(匿名方法)(二)的相关文章

C# 2.0的Specification(匿名方法)(二)

21.7委托实例相等性 如下规则适用由匿名方法委托实例的相等运算符(§7.9.8)和object.Equals方法产生的结果. l 当委托实例是由具有相同被捕获外部变量集合的语义相同的匿名方法表达式计算而产生时,可以说(但不是必须)它们相等. l 当委托实例由具有语义不同的匿名方法表达式,或具有不同的被捕获外部变量集合时,它们决不相等. 21.8明确赋值 匿名方法参数的明确赋值状态与命名方法是相同的.也就是,引用参数和值参数被明确的赋初值,而输出参数不用赋初值.并且,输出参数在匿名方法正常返回之

C# 2.0 Specification(匿名方法)(一)

21匿名方法21.1.匿名方法表达式匿名方法表达式(anonymous-method-expression)定义了匿名方法(anonymous method),它将计算为引用该方法的一个具体值. l primary-no-array-creation-expression(基本非数组创建表达式:)-anonymous-method-expression(匿名方法表达式) l anonymous-method-expression:delegate anonymous-method-signatu

C# 2.0的Specification(匿名方法)(一)

21匿名方法21.1.匿名方法表达式 匿名方法表达式(anonymous-method-expression)定义了匿名方法(anonymous method),它将计算为引用该方法的一个具体值. l primary-no-array-creation-expression(基本非数组创建表达式:) - anonymous-method-expression(匿名方法表达式) l anonymous-method-expression: delegate anonymous-method-sig

C# 2.0 Specification(迭代器)(二)

22.4 yield 语句yield语句用于迭代器块以产生一个枚举器对象值,或表明迭代的结束. embedded-statement:(嵌入语句)...yield-statement(yield语句) yield-statement:(yield 语句)yield return expression ;yield break ; 为了确保和现存程序的兼容性,yield并不是一个保留字,并且 yield只有在紧邻return或break关键词之前才具有特别的意义.而在其他上下文中,它可以被用作标识

对C# 2.0中匿名方法的怀疑分析

一.简介 所有的方法都使用一个来自于相同集合的元素的子集.在C# 2.0中,可选元 素集将会继续增长.从历史上看-除了C++内联方法之外-方法都要求有一个名字 .一个返回类型和一个方法体.而且可选择地,方法可以使用存取修饰符和一个 参数列表.在C# 2.0中,方法名已经从必需的变成了可选的. C# 2.0(一般就代表.NET)引入了匿名方法.一个匿名方法可以被用在任何使 用代理且该代理被定义为内联的情况下,它不需要方法名,而具有可选的参数和 一个方法体. 为了使用匿名方法,你需要了解什么是代理.

匿名方法,Lambda表达式,高阶函数

原文:匿名方法,Lambda表达式,高阶函数 匿名方法 c#2.0引入匿名方法,不必创建单独的方法,因此减少了所需的编码系统开销. 常用于将委托和匿名方法关联,例如 1. 使用委托和方法关联: this.btnRefresh.Click += new System.EventHandler(this.btnRefresh_Click);private void btnRefresh_Click(object sender, EventArgs e){    BindData();} 2. 使用委

最简单的匿名方法与委托示例

 1using System; 2 3namespace ConsoleApplication1 4{ 5    class Program 6    { 7        delegate string delegateTest(string val); 8 9        //被委托调用的方法10        static string Test(string str)11        {           12            str += " 这是传统1.0的方法"

C# 2.0:使用匿名方法、迭代程序和局部类来创建优雅的代码

程序|创建 本文基于 Microsoft Visual Studio 2005 的预发布版本,它以前的代码名称为"Whidbey".此处所包含的任何信息都可能会改变. 本文讨论: • 遍历集合 • 跨文件类定义 • 与委托一起使用的匿名方法 • Visual Studio 2005 中的其他 C# 新功能 本文使用下列技术: • C# 和 Visual Studio 可以在此下载代码: • C20.exe (164KB) 本页内容 迭代程序 迭代程序实现 递归迭代 局部类型 匿名方法

Visual C# 2.0匿名方法揭密

visual 匿名方法基础 匿名方法是C#2.0的一个新的语言特性.本文的主要内容是提供给读者关于匿名方法的内部实现和工作方式的一个更好的理解.本文无意于成为匿名方法的完全语言特性参考. 匿名方法允许我们定义委托对象可以接受的代码块.这个功能省去我们创建委托时想要传递给一个委托的小型代码块的一个额外的步骤.它也消除了类代码中小型方法的混乱.让我们看看:比方说,我们有一个字符串集合命名为MyCollection.这个类有一个方法:获得集合中满足用户提供的过滤准则的所有项,调用者决定在集合中的一个特