(原题链接:http://community.csdn.net/Expert/topic/3918/3918948.xml?temp=.2294428)
编写一个控制台应用程序,完成下列功能,并回答提出的问题。
1.创建一个类A,在构造函数中输出“A”,再创建一个类B,在构造函数中输出“B”。
2.从A继承一个名为C的新类,并在C内创建一个成员B。不要为C创建构造函数。
3.在Main方法中创建类C的一个对象,写出运行程序后输出的结果。
4.如果在C中也创建一个构造函数输出“C”,整个程序运行的结果又是什么?
要简单回到这道题是简单的,我的答案是:
2
3class A
4{
5 public A()
6 {
7 Console.WriteLine("A");
8 }
9}
10
11class B
12{
13 public B()
14 {
15 Console.WriteLine("B");
16 }
17}
18
19class C : A
20{
21 public C()
22 {
23 Console.WriteLine("C");
24 }
25 B b = new B();
26}
27
28class Test
29{
30 public static void Main()
31 {
32 C c = new C();
33
34 }
35}
跟作者的参考答案是一致的,但是作者在结帖的时候并没有回答有人提出的一个问题:问什么是这样的输出顺序呢?
我就自己想了一下。首先看输出解结果“B”是调用B类的实例构造函数输出的,“A”是调用A类的构造函数输出的,“C”调用C类的构造函数输出的。
其中B类作为一个字段出现在C类中,而C类继承A类。那么,我猜,在主函数中构造C对象,首先初始化C类的字段,然后调用祖先类的构造函数,再实现C类自己构造函数中的代码,这样就解释了这种顺序。
那么这种猜想成不成立呢?
首先初始化C类的字段可能会有些疑惑,为什么非要先初始化类的字段呢?这里就扯到另外一个问题:字段的内联初始化是在哪里初始化的(这里的内联是指声明字段时同时进行初始化赋值,比如public int i = 1,而不是public in i),在《.NET框架程序设计》9.1节实例构造器中有解释,这里我只说结果,就是内联初始化实际上是在类型的构造器中完成的。那么这样看来,初始化B类对象b跟C中的Console.WriteLine()方法都发生在C类的构造器中,其实调用父类A的构造器也发生在C的构造器,至于为什么非要把内联初始化字段放在最前面,我就不清楚了,还请大家指点(可能有什么先搞定字段,再搞方法的规则吧,哈哈)。不过这个顺序是没错的,有IL作证:
2 instance void .ctor() cil managed
3{
4 // 代码大小 28 (0x1c)
5 .maxstack 2
6 IL_0000: ldarg.0
7 IL_0001: newobj instance void B::.ctor()
8 IL_0006: stfld class B C::b
9 IL_000b: ldarg.0
10 IL_000c: call instance void A::.ctor()
11 IL_0011: ldstr "C"
12 IL_0016: call void [mscorlib]System.Console::WriteLine(string)
13 IL_001b: ret
14} // end of method C::.ctor
以上是C构造函数的IL代码,我们可以清楚地看到首先创建可B的实例b,当然同时调用了B的构造函数,然后把b值存为C的对象的一个字段,然后调用A的构造函数,然后在调用WriteLine方法。
可能会有人对B b = new B()是内联初始化不太理解,那么我再添加以行代码int i = 1;到后面,使C类变为:
{
public C()
{
Console.WriteLine("C");
}
B b = new B();
int i = 1;
}
那么IL就变成:
instance void .ctor() cil managed
{
// 代码大小 35 (0x23)
.maxstack 2
IL_0000: ldarg.0
IL_0001: newobj instance void B::.ctor()
IL_0006: stfld class B C::b
IL_000b: ldarg.0
IL_000c: ldc.i4.1
IL_000d: stfld int32 C::i
IL_0012: ldarg.0
IL_0013: call instance void A::.ctor()
IL_0018: ldstr "C"
IL_001d: call void [mscorlib]System.Console::WriteLine(string)
IL_0022: ret
} // end of method C::.ctor
你看,这回比较清楚了吧,字段的初始化是在前面,同时说明了一个问题,就是,要避免内联初始化字段,这样会增加代码尺寸,把初始化放构造器中去进行, 这里不再细述。