构造函数链接顺序

时间:2011-01-05 22:16:25

标签: c#

如果使用以下语法链接构造函数调用:

public frmConfirm(): this(1)

何时调用重载的构造函数?此外,有人可以确认如果类是一个表单,那么在两个构造函数中调用InitializeComponent()会产生问题吗?

3 个答案:

答案 0 :(得分:44)

链式构造函数将在定义构造函数的主体之前立即调用。生成的IL序列是另一个构造函数的立即call,后跟从构造函数中的语句生成的IL。

因此,如果链接到另一个构造函数并且构造函数调用InitializeComponent(),则调用构造函数不应该调用此方法。

例如,给定此示例类:

class Foo {
    public int A, B;

    public Foo() : this(1) {
        B = 2;
    }

    public Foo(int a) {
        A = a;
    }
}

这是生成的IL:

  .class private auto ansi beforefieldinit Foo
        extends [mscorlib]System.Object
  {
    .field  public  int32 A
    .field  public  int32 B

    // method line 1
    .method public hidebysig  specialname  rtspecialname
           instance default void '.ctor' ()  cil managed
    {
        .maxstack 8
        IL_0000:  ldarg.0
        IL_0001:  ldc.i4.1
        IL_0002:  call instance void class Foo::'.ctor'(int32)
        IL_0007:  ldarg.0
        IL_0008:  ldc.i4.2
        IL_0009:  stfld int32 Foo::B
        IL_000e:  ret
    } // end of method Foo::.ctor

    // method line 2
    .method public hidebysig  specialname  rtspecialname
           instance default void '.ctor' (int32 a)  cil managed
    {
        .maxstack 8
        IL_0000:  ldarg.0
        IL_0001:  call instance void object::'.ctor'()
        IL_0006:  ldarg.0
        IL_0007:  ldarg.1
        IL_0008:  stfld int32 Foo::A
        IL_000d:  ret
    } // end of method Foo::.ctor

  } // end of class Foo

请注意,no-arg构造函数在将2分配给B字段之前调用另一个构造函数。

答案 1 :(得分:11)

首先调用this(1)构造函数。

至于你的第二个问题,由于InitializeComponent和表单继承的其他问题,我建议你使用组合而不是继承。

答案 2 :(得分:7)

在这样的问题上寻找答案的地方是C# Language Specification。在 10.11.1,构造函数初始值设定项一节中,您可以阅读(强调是我的):

  

所有实例构造函数(除了   隐含地用于类对象的那些   包括对另一个人的调用   实例构造函数立即   之前构造函数体。

进一步阅读表明:

  • 如果构造函数具有base(arguments)形式的实例构造函数初始值设定项,则将调用直接基类的构造函数。
  • 如果构造函数具有this(argument)形式的实例构造函数初始值设定项,则将调用类本身的构造函数。
  • 如果未提供实例构造函数初始值设定项,则会自动添加base()