前一阵子我参加了一次笔试,其中有一道选择题让我印象深刻,是这样的:
实例化一个X类型对象时所执行的顺序:
A.调用X类型构造函数,调用X类型基类的构造函数,调用X类型内部字段的构造函数
B.调用X类型内部字段的构造函数,调用X类型基类的构造函数,调用X类型构造函数
C.调用X类型基类的构造函数,调用X类型构造函数,调用X类型内部字段的构造函数
D.调用X类型基类的构造函数,调用X类型内部字段的构造函数,调用X类型构造函数
我觉的这道题出得很没水平。在C++的世界里,我会毫不犹豫的选D。但是,由于C#引入了字段初始化器,所以选什么答案完全依赖于类具体是如何设计的。好吧,我们今天就来谈谈C#在类型实例化时都有哪些步骤。
首先我们都知道,对于类对象,在执行构造函数之前,我们需要使用关键字new来为新实例分配内存。new可以根据对象的类型来为其在堆上分配足够的空间,并且将这个对象的所有字段都设为默认值。也就是说,CLR会把该对象的所有引用类型字段设为null,而把值类型字段的所有底层二进制表示位设为0(本质上来说,不论是将值类型或引用类型字段初始化为“默认值”,其实都是把他们底层的数据位设为0)。这是任何类对象实例化的第一步。
我们暂且先不考虑对象有指定基类的情况,先看看下面的代码吧:
class
MyClass{
static MyClass()
{
Console.WriteLine( " 静态构造函数被调用。 " );
}
private static Component staticField = new Component( " 静态字段被实例化。 " );
private Component instanceField
static MyClass()
{
Console.WriteLine( " 静态构造函数被调用。 " );
}
private static Component staticField = new Component( " 静态字段被实例化。 " );
private Component instanceField