4

当我尝试 C# 4.0 中的新特性“可选参数”时,我发现了这个有趣的事情。

我知道在 C# 4.0 中有两种使用“可选参数”的方法:

static void TestMethod(int parameter = 5) { }
static void TestMethod2([Optional, DefaultParameterValue(5)]int parameter) { }

现在,如果我编译这段代码,然后使用 IL Dasm 查看程序集,那么我会看到在 IL Dasm 的“MetaInfo”窗口中,有一个对 OptionalAttribute 的类型引用,如下所示:

Token:             0x01000002
ResolutionScope:   0x23000001
TypeRefName:       System.Runtime.InteropServices.OptionalAttribute
MemberRef #1 (0a000001)
-------------------------------------------------------
    Member: (0a000001) .ctor: 
    CallCnvntn: [DEFAULT]
    hasThis 
    ReturnType: Void
    No arguments.

但是没有 DefaultParameterValueAttribute 的迹象。这是为什么?

其实我认为这两个属性都不应该在这里,因为编译器对它们的处理不同,它们有自己的标志值。为了解释我的意思,请看一下:

    Method #2 (06000002) 
-------------------------------------------------------
    MethodName: TestMethod (06000002)
    Flags     : [Private] [Static] [HideBySig] [ReuseSlot]  (00000091)
    RVA       : 0x00002053
    ImplFlags : [IL] [Managed]  (00000000)
    CallCnvntn: [DEFAULT]
    ReturnType: Void
    1 Arguments
        Argument #1:  I4
    1 Parameters
        (1) ParamToken : (08000002) Name : parameter flags: [Optional] [HasDefault]  (00001010) Default: (I4) 5

Method #3 (06000003) 
-------------------------------------------------------
    MethodName: TestMethod2 (06000003)
    Flags     : [Private] [Static] [HideBySig] [ReuseSlot]  (00000091)
    RVA       : 0x00002056
    ImplFlags : [IL] [Managed]  (00000000)
    CallCnvntn: [DEFAULT]
    ReturnType: Void
    1 Arguments
        Argument #1:  I4
    1 Parameters
        (1) ParamToken : (08000003) Name : parameter flags: [Optional] [HasDefault]  (00001010) Default: (I4) 5

那是这两种方法的元数据。我们可以看到每个代码段的最后一行已经存储了默认的参数值,那为什么还要引用 OptionalAttribute 呢?

4

1 回答 1

3

这可以被认为是编译器中的一个错误。如果您仔细检查程序集,则没有任何内容引用 OptionalAttribute TypeRef。

如果您没有明确添加 TestMethod2,TypeRef 将不会出现在生成的程序集中。这只是表明该属性是在第一遍中添加的,然后是删除的。它还表明编译器对 DefaultParameterValueAttribute 的处理方式不同。

最终,这实际上是一个小的实现细节,例如,Mono 的 C# 编译器不会为生成的程序集中的任一属性生成 TypeRef。

于 2011-04-13T06:40:02.790 回答