构造函数这玩意也是面试官经常会问到的东西,我们知道一个类的状态初始化就全靠它了,下面提几个小问题。
Q:我看到Web项目里面通常都有一个BasePage页面,其中构造函数里面做了权限验证,
请问为什么要这么做。
A: 既然这么做了,那么设计者肯定就知道了一个原则就是实例构造器中,父类构造器先于子类执行,那么这个先执行就可
以做很多有意思的事情,比如你说的权限验证,可能有人会问为什么要先于子类执行,刚才我也说了,构造函数是用于
初始化本类状态的,这也叫“先扫屋子再请客”的道理吧~,然后风雨兼程的回溯到Object的ctor中。好了,现在来回答
你的问题,先把这个问题简化一下,就不用web项目来演示了。
public class Program
{
static void Main(string[] args)
{
var b = new Bird();
Console.Read();
}
}
public class Animal
{
public Animal()
{
Console.WriteLine("running first... i'm animal, all must be running after me.");
}
}
public class Bird : Animal
{
public Bird()
{
Console.WriteLine("the next... i'm a cute bird.");
}
}
可以看到,确实Animal于Bird先执行,可能有人会问,那Animal调用的是谁的ctor呢?当然就是Object了,我们可以看看IL:
Q: 既然你说构造函数用于初始化类的初始状态,那么请问下,我现在有一串这样的
json:{"Name":"smart","Age":2}。实体类如下图,请问我的json序列化后,Age=?
[Serializable]
public class Bird
{
private string name = "smart";
public string Name
{
get { return name; }
set { name = value; }
}
private int age = 2;
public int Age
{
get { return age; }
set { age = value; }
}
public Bird()
{
Age = 5;
}
}
A:其实这个问题的关键在于,反序列化的时候是否会调用构造函数的问题,先来通过Reflector看下源码,发现并没有代码,觉得
有点小奇怪,就用ILSpy反编译一下,同样也没有代码,不可否认,底层肯定是有代码执行的,要么反编译不出来,要么clr用了
其他的方式,反正我们看不到实现源代码,如果大家有什么好意见可以帮帮我,谢谢了
Reflector截图:
ILSpy截图:
据资料说,里面最后调用了FormatterServices里面的GetSafeUninitializedObject来分配对象内存,分配完后就不走构造器
了,所以针对这个问题,只能记住了。最后为了验证一下,执行完代码之后确实没有走构造函数。
Q:我知道引用类型可以用构造器,那值类型为什么不能定义无参构造器?
A:这个问题问的好,编译器会根据性能考虑,不会调用值类型的构造函数,即使你强制的new一下也不会执行
namespace Sample
{
public class Program
{
static void Main(string[] args)
{
Point point = new Point();
}
}
public struct Point
{
public int Age;
}
}
不过除非你自己显示定义有参构造函数,而且值类型有个特点就是读取之前必须初始化,否则会编译不通过。
Q:请问在类构造函数中能做单例吗?
A: 能不能做,就要看类构造器的特征了,我们知道类构造器跟实例构造器一样,他是用来初始化静态字段的,线程
访问类构造器的时候内部会进行加锁处理,所以多个线程同时访问的时候,只会有一个线程执行了类构造器,所以
确实可以。