加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 编程开发 > asp.Net > 正文

深入理解C# 3.x的新特性(2):Extension Method[下篇]

发布时间:2020-12-16 09:09:05 所属栏目:asp.Net 来源:网络整理
导读:四、Extension Method的本质 通过上面一节的介绍,我们知道了在C#中如何去定义一个Extension Method:它是定义在一个 Static class 中的、第一个Parameter标记为 this 关键字的 Static Method 。在这一节中,我们来进一步认识Extension Method。 和C# 3.0的

四、Extension Method的本质

通过上面一节的介绍,我们知道了在C#中如何去定义一个Extension Method:它是定义在一个Static class中的、第一个Parameter标记为this关键字的Static Method。在这一节中,我们来进一步认识Extension Method。

和C# 3.0的其他新特性相似,Extension Method仅仅是C#这种.NET Programming Language的新特性而已。我们知道,C#是一种典型的编译型的语言,我们编写的Source Code必须先经过和C# Compiler编译成Assembly,才能被CLR加载,被JIT 编译成Machine Instruction并最终被执行。C# 3.0的这些新的特性大都影响Source被C# Compiler编译成Assembly这个阶段,换句话说,这些新特仅仅是Compiler的新特性而已。通过对Compiler进行修正,促使他将C# 3.0引入的新的语法编译成相对应的IL Code,从本质上看,这些IL Code 和原来的IL并没有本质的区别。所有当被编译生成成Assembly被CLR加载、执行的时候,CLR是意识不到这些新的特性的。

从Extension Method的定义我们可看出,Extension Method本质上是一个Static Method。但是我们往往以Instance Method的方式进行调用。C# Compiler的作用很明显:把一个以Instance Method方式调用的Source Code编译成的于对应于传统的Static Method调用的IL Code

虽然Extension Method本质上仅仅是一个Static Class的Static Method成员,但是毕竟和传统的Static Method有所不同:在第一个Parameter前加了一个this关键字。我们现在来看看他们之间的细微的差异。我们先定义一个一般的Static Method:

public?static?Vector?Adds(Vector?v,?Vector?v1)
{
??
returnnew?Vector?{?X?=?v.X?+?v1.X,?Y??v.Y??v1.Y?};
}

注:Vector的定义参见《深入理解C# 3.0的新特性(2):Extension Method - Part I》。

我们来看看通过Compiler进行编译生成的IL:

.method?private?hidebysig?void??Main(string[]?args)?cil?managed
{
??.entrypoint
??
//?Code?size???????50?(0x32)
??.maxstack??2
??.locals?init?([
0]?class?Artech.ExtensionMethod.Vector?v,
???????????[
1?Artech.ExtensionMethod.Vector?'<>g__initLocal0)
??IL_0000:??nop
??IL_0001:??newobj?????instance?
?Artech.ExtensionMethod.Vector::.ctor()
??IL_0006:??stloc.

??IL_0007:??ldloc.

??IL_0008:??ldc.r8?????
.
??IL_0011:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_X(float64)
??IL_0016:??nop
??IL_0017:??ldloc.

??IL_0018:??ldc.r8?????
.
??IL_0021:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_Y(float64)
??IL_0026:??nop
??IL_0027:??ldloc.

??IL_0028:??stloc.

??IL_0029:??ldloc.

??IL_002a:??ldloc.

??IL_002b:??call???????
?Artech.ExtensionMethod.Vector?Artech.ExtensionMethod.Extension::Adds(?Artech.ExtensionMethod.Vector,
?Artech.ExtensionMethod.Vector)
??IL_0030:??stloc.

??IL_0031:??ret
}?
?end?of?method?Program::Main

对了解IL的人来说,对上面的IL code应该很容易理解。

我们再来看看对于通过下面的方式定义的Extension Method:

?Extension

????

对于得IL如下:

?Artech.ExtensionMethod.Vector?

Adds(

?Artech.ExtensionMethod.Vector?v1)?cil?managed

{

??.custom?instance?
?[System.Core]System.Runtime.CompilerServices.ExtensionAttribute::.ctor()??(?0100?)?

??
?Code?size???????53?(0x35)

3

??.locals?init?([
,

???????????[
?Artech.ExtensionMethod.Vector?CS$$0000)

??IL_0000:??nop

??IL_0001:??newobj?????instance?
?Artech.ExtensionMethod.Vector::.ctor()

??IL_0006:??stloc.

??IL_0007:??ldloc.

??IL_0008:??ldarg.

??IL_0009:??callvirt???instance?float64?Artech.ExtensionMethod.Vector::get_X()

??IL_000e:??ldarg.

??IL_000f:??callvirt???instance?float64?Artech.ExtensionMethod.Vector::get_X()

??IL_0014:??add

??IL_0015:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_X(float64)

??IL_001a:??nop

??IL_001b:??ldloc.

??IL_001c:??ldarg.

??IL_001d:??callvirt???instance?float64?Artech.ExtensionMethod.Vector::get_Y()

??IL_0022:??ldarg.

??IL_0023:??callvirt???instance?float64?Artech.ExtensionMethod.Vector::get_Y()

??IL_0028:??add

??IL_0029:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_Y(float64)

??IL_002e:??nop

??IL_002f:??ldloc.

??IL_0030:??stloc.

??IL_0031:??br.s???????IL_0033

??IL_0033:??ldloc.

??IL_0034:??ret

}
?end?of?method?Extension::Adds

通过比较,我们发现和上面定义的一般的Static Method生成的IL唯一的区别就是:在Adds方法定义最开始添加了下面一段代码:

.custom?instance??)?

这段添加的IL代码很明显,就是在Adds方法上添加一个Customer Attribute:System.Runtime.CompilerServices.ExtensionAttribute。ExtensionAttribute具有如下的定义:

[AttributeUsage(AttributeTargets.Method?|?AttributeTargets.Class??AttributeTargets.Assembly)]

sealed?ExtensionAttribute?:?Attribute

{

}

所以下面Extension Method的定义

{

????????????
?Vector?

;

}

和下面的定义是等效的

[ExtensionAttribute]

{

????????????
?Vector?


但是,System.Runtime.CompilerServices.ExtensionAttribute和其他Custom Attribute不一样,因为它是为了Extension Method的而定义的,我们只能通过添加this Key word的语法来定义Extension Method。所以当我们将System.Runtime.CompilerServices.ExtensionAttribute直接运用到Adds方法会出现下面的Compile Error:

Do?not?use?System.Runtime.CompilerServices.ExtensionAttribute.?Use?the??keyword?instead.

上面我们比较了Extension Method本身IL和一般Static Method IL,现在我们看看当我们以Instance Method方式调用Extension Method的IL。假设我们通过下面的方式调用Adds。 

?Program

????

{

????????
?Main([]?args)

????????

{

?var?v?
?Vector?

?};

???????????v?
?v.Adds(v);

????????}

下面是Main Method的IL:

[]?args)?cil?managed

{

??.entrypoint

??

??IL_0008:??ldc.r8?????
.

??IL_0011:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_X(float64)

??IL_0016:??nop

??IL_0017:??ldloc.

??IL_0018:??ldc.r8?????
.

??IL_0021:??callvirt???instance?
?Artech.ExtensionMethod.Vector::set_Y(float64)

??IL_0026:??nop

??IL_0027:??ldloc.

??IL_0028:??stloc.

??IL_0029:??ldloc.

??IL_002a:??ldloc.

??IL_002b:??call???????

?Artech.ExtensionMethod.Vector)

??IL_0030:??stloc.

??IL_0031:??ret

}
?end?of?method?Program::Main

通过上面的IL,我们看到调用的是Artech.ExtensionMethod.Extension的Adds方法。

IL_002b:??call??Artech.ExtensionMethod.Vector)

通过对IL的分析,我们基本上看出了Extension Method的本质。我们再来简单描述一下对Compiler的编译过程:当Compiler对Adds方法的调用进行编译的过程的时候,它必须判断这个Adds方式是Vector Type的成员还是以Extension Method的方式定义。Extension Method的优先级是最低的,只有确定Vector中没有定义相应的Adds方法的时候,Compiler才会在引用的Namespace中查看这些Namespace中是否定义有对应的Adds Extension Method的Static Class。找到后作进行相应的编译,否则出现编译错误。

五、一个完整的Extension Method的Sample

在介绍了Extension Method的本质之后,我们通过一个相对完整的Sample进一步了解Extension Method的运用,通过这个Sample,我们还可以粗略了解LINQ的原理。

C# 3.0为LINQ定义了一系列的Operator:select, from,where,orderby...,促使我们按照OO的方式来处理各种各样的数据,比如XML,Relational DB Data,C#中IEnumeratable<T> Object。比如:

var?names??List<>?

{?"Tom?CruiseTom?HanksAl?PacinoHarrison?Ford;

var?result?
?names.Where(name?=>?name.StartsWith(Tom));

foreach(var?name?in?result)

{

??????Console.WriteLine(name);

}

我们通过上面的Code,从一系列的姓名列表中("Tom Cruise","Tom Hanks","Al Pacino","Harrison Ford")筛选名字(First Name)为Tom的姓名。通过Where Operator,传入一个以Lambda Expression表示的筛选条件(name => name.StartsWith("Tom"))。Where Operator就是通过Extension Method的方式定义的。

在这里提供的Sample就是定义一个完成Where Operator相同功能的Operator,我们把这个Operator起名为When

using?System;

?System.Collections.Generic;

?System.Linq;

?System.Text;

?System.Collections;


namespace?Artech.ExtensionMethod

{

????
delegate?TResult?FunctionTparam,?TResult(Tparam?param);


????
?Extension

????

?IEnumerableTSource?When(?source,?FunctionTSource,1)">bool?predicate)

????????

{

????????????
?WhenEnumerator(source,?predicate);

????????}
??


????}


????
?:?IEnumerable

????

?_source;

????????
?Function?_predicate;

????????
?IEnumerator?_sourceEnumerator;


????????
?WhenEnumerator(IEnumerable?predicate)

????????

._source??source;

????????????
._predicate??predicate;

????????????
._sourceEnumerator?._source.GetEnumerator();

????????}


????????
IEnumerable?Members

我们来看看我们新的LINQ Operator:When的定义。我首先定义了一个Generic Delegate:Function。实际上他定义了一个一元函数y?=?f(x),TParam和TResult为参数和返回值得类型。?

(Tparam?param);

接着在Static Class Extesnion中定义了Extension Method:When。该方法包含两个参数,其中一个是执行筛选的数据源,另一个是用于判断数据源每个对象是否满足你所定义的筛选条件的断言。返回一个我们自定义的、实现了IEnumerable的WhenEnumerator对象。

?Extension

????

?predicate)

????????

?

????}

WhenEnumerator的定义是实现When Extension Method的关键,我们现在着重来介绍它的具体实现。WhenEnumerator实现了Interface Enumerable<T>,为了简单,我们也它对应的Enumerator的实现也定义在同一个Class中,所以WhenEnumerator实现了两个Interface:IEnumerable<TSource>,IEnumerator<TSource>。?

以下3个成员分别代表:用于执行筛选的数据源、用于判断是否满足筛选条件的断言以及数据源的Enumerator对象。

?_source;

?_predicate;

?_sourceEnumerator;

通过返回一个WhenEnumerator对象,实现了IEnumerable<TSource>的GetEnumerator()方法。?

?????????GetEnumerator()

????????

._predicate);

????????}

对于另一个Interface IEnumerator<TSource>,直接调用数据源的Enumerator的同名方法实现了Current,和Reset()。对于MoveNext()则通过如下的方式实现:把当前的位置设置在下一个满足筛选条件的Element上

?MoveNext()

????????

._sourceEnumerator.MoveNext())

????????????

._sourceEnumerator.Current))

????????????

._sourceEnumerator.MoveNext())

????????????????

;

????????}

到现在为止,这个新的LINQ Operator被创建,现在我们可以按照使用Where operator的方式来调用When。

我们可以通过Delegate的方式来使用When Operator:

?Program

????

?Main()

????????

{

????????????var?names?
?

;

????????????var?result?
?names.When(?name)?

);?});

????????????
?(var?name??result)

????????????

{

????????????????Console.WriteLine(name);

????????????}

输出结果:

Tom?Cruise

Tom?Hanks

我们也可以通过Lambda Expression的方式来使用When Operator:

?Main()

????????

{

????????????var?names?
?

;

????????????var?result?
?names.When(namename.StartsWith());

????????????
?result)

????????????


????????}

显然这种方式更简洁。?

Deferred Evaluation

对于LINQ,有一个非常重要的特征:Deferred Evaluation。在了解这个特征之前,我们来看一个例子:

?Main()

{

????????????var?names?
?

;

????????????var?result1?
));

????????????names[
Stephen?Chou;

????????????var?result2?
?names.When(name?));



????????????
?result1)

????????????

?result2)

????????????


运行程序,你会发现两个foreach loop显示的结果都是一样的:Tom Hanks。为什么result1实在第一个Element被改动之前返回的,但我们最终输出的结果却反映的是改动之后的数据源。通过我们上面的定义,你很容易得到答案。在这里我要说的是LINQ的一个重要的特性Deferred Evaluation:在调用Operator的时候并不会有任何的任何数据获取的过程,这个阶段的任务是创建一个同于获取数据的表达式。只要你真正所用到这个数据的时候,采用重数据源中通过你构建的表达式通过查询获取数据。

C# 3.x相关内容:
[原创]深入理解C# 3.x的新特性(1):Anonymous Type
[原创]深入理解C# 3.x的新特性(2):Extension Method - Part I
[原创]深入理解C# 3.x的新特性(2):Extension Method - Part II
[原创]深入理解C# 3.x的新特性(3):从Delegate、Anonymous Method到Lambda Expression
[原创]深入理解C# 3.x的新特性(4):Automatically Implemented Property
[原创]深入理解C# 3.x的新特性(5):Object Initializer 和 Collection Initializer

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读