1、静态构造函数  

  在引入本文的主题之前,我们先来铺垫一下吧,看看静态构造函数的概念及用途。

  C#中允许创建无参数构造函数,该函数仅执行一次。它一般被用来初始化静态字段。CLR不能保证在某个特定时刻执行静态构造函数,同时也不保证不同类的静态构造函数按照什么顺序执行,但保证它仅执行一次,即在应用程序创建该类的第一个实例或访问该类的任何静态成员之前。

  注意,静态构造函数不允许有访问修饰符,且不接受任何参数,这是因为其他代码没有权利调用它,它的调用执行总是被CLR接管的!另外,一个类只能有一个静态构造函数。

2、字段及构造函数的初始化顺序

  首先,我们需要明确的一点是:字段和构造函数均分为静态和实例两大类。其中,静态的属于类型本身,仅有一份;而实例的属于创建的实例对象,可有多份。静态的总是先于实例的被初始化。

  在初始化一个C#对象时,如果我们晓得字段和构造函数被初始化的顺序,就能够防止一些错误发生,比如引用还未初始化的字段等,同时对对象初始化构造的过程有更深刻的认识。

  首先,我们先给出初始化先后顺序的结论,然后再以例子来佐证:

  1、继承类静态字段

  2、继承类静态构造函数

  3、继承类实例字段

  4、基类静态字段

  5、基类静态构造函数

  6、基类实例字段

  7、基类实例构造函数

  8、继承类实例构造函数

  我们知道对于继承层次结构的类来说,首先调用基类的构造函数,然后调用继承类的构造函数。

  下面我们以一个控制台程序的执行来说明。

  

class Program
{
static void Main(string[] args)
{
Derived d = new Derived();
Console.ReadLine();
}
}
class Base
{
public Base()
{
Console.WriteLine("基类实例构造器");
this.m_Field3 = new ShowMessage("基类实例字段3");
this.Virtual();
}
static Base()
{
Console.WriteLine("基类静态构造器");
}
private ShowMessage m_Field1 = new ShowMessage("基类实例字段1");
private ShowMessage m_Field2 = new ShowMessage("基类实例字段2");
private ShowMessage m_Field3;
static private ShowMessage s_Field1 = new ShowMessage("基类静态字段1");
static private ShowMessage s_Field2 = new ShowMessage("基类静态字段2");
virtual public void Virtual()
{
Console.WriteLine("基类实例虚方法");
}
}
class Derived : Base
{
public Derived()
{
Console.WriteLine("继承类实例构造器");
this.m_Field3 = new ShowMessage("继承类实例字段3");
}
static Derived()
{
Console.WriteLine("继承类静态构造器");
}
private ShowMessage m_Field1 = new ShowMessage("继承类实例字段1");
private ShowMessage m_Field2 = new ShowMessage("继承类实例字段2");
private ShowMessage m_Field3;
static private ShowMessage s_Field1 = new ShowMessage("继承类静态字段1");
static private ShowMessage s_Field2 = new ShowMessage("继承类静态字段2");
override public void Virtual()
{
Console.WriteLine("继承类实例虚方法");
}
}
class ShowMessage
{
public ShowMessage(string msg)
{
Console.WriteLine(msg);
}
}
}

  上面的程序中,Drived子类和Base基类均包含静态和实例构造器以及静态和实例字段。其中,实例字段m_Field1和m_Field2在字段定义时初始化,而字段m_Field3在实例构造起中初始化。同时,在Base基类的构造器中调用了一个Virtual虚方法,这一步是为了说明在构造器中调用虚方法存在的潜在风险,实际开发中应避免这样做。

  从上面的执行过程中,可以看出,继承类静态字段和静态构造器首先初始化,很明显,静态的东西是类型本身固有的东西,所以,在初始化一个实例对象之前,首先要保证静态的被初始化。而无论静态还是实例字段,它们的初始化过程总是优先于相对应的构造函数中的其余代码的初始化。

  在继承类的静态字段和构造器执行完毕之后,接着执行实例字段1和2的初始化,这两个字段属于在定义时初始化,先于实例构造函数的调用,这是因为要防止在构造函数中调用未初始化的字段(构造器若调用了一些方法,而这些方法访问了未初始化的字段就会发生这种情况)。我们之所以将字段的初始化提前到构造函数执行之前,目的就在于避免null reference exceptions,这在大多数情况下是更好的选择,但这种方式也有缺点,即:在调试对象初始化部分代码时,若想进入构造函数必须先经过一系列的字段成员初始化代码,尤其在继承层次复杂时,这种混乱更加明显。在这两个字段初始化之后,开始实例构造函数的调用,由于存在继承关系,故先调用基类的构造函数。

  在调用基类的构造函数之前,首先要初始化基类的静态字段和静态构造器,原因同继承类。然后初始化基类的实例字段1和2,接着调用基类实例构造器并初始化实例字段3。接着在基类的实例构造器中调用了Virtual虚方法。这里需要注意的是:这时候继承类的实例构造函数还没有被执行,如果此虚方法中需要使用已初始化的字段,比如本例中的字段3,那么,程序将会导致错误。这提示我们在编程时应尽量避免在构造函数中调用虚方法,而应在对象初始化完成之后再调用虚方法。

3、字段初始化器和构造函数初始化字段的差异

  两者的差异主要有两方面:

  1、用字段初始化器初始化字段就不能使用this,因为此时构造函数还未调用。

  2、第二个差异点就在于存在继承层次结构时,因为在这种情形下字段初始化器和构造函数的执行顺序不一致,前者的执行顺序是由继承类到基类,后者是由基类到继承类。

  最后,有一些情形需要特别注意,比如下面的例子,请读者自行思考。

class Base
{
private readonly object objectA = new object(); // 第二执行
private readonly object objectB; public Base()
{
this.objectB = new object(); // 第三执行
}
} class Derived : Base
{
private object objectC = new object(); // 首先执行
private object objectD; public Derived()
{
this.objectD = new object(); // 第四执行
}
}  

对象初始化的完整过程(C#)的更多相关文章

  1. JAVA 对象初始化的过程

    对象初始化的过程例:Student S    =    new Student();1.因为new Student()用到了Student类,所以会把它从硬盘上加载进入内存2.如果有static静态代 ...

  2. Java父类子类的对象初始化过程

    摘要 Java基本的对象初始化过程,子类的初始化,以及涉及到父类和子类的转化时可能引起混乱的情况. 1. 基本初始化过程: 对于一个简单类的初始化过程是: static 修饰的模块(static变量和 ...

  3. Java类与对象初始化的过程(一道经典的面试题)

    本文不再以ClassLoader的视角解释这些问题. 首先,Java代码有个特点,就是成员变量可以在前面的方法中使用,在后面定义.这一特性,很多人说Java了不起,可是为什么呢?Java为何能够这样呢 ...

  4. java基础课程笔记 static 主函数 静态工具类 classpath java文档注释 静态代码块 对象初始化过程 设计模式 继承 子父类中的函数 继承中的构造函数 对象转型 多态 封装 抽象类 final 接口 包 jar包

    Static那些事儿 Static关键字 被static修饰的变量成为静态变量(类变量) 作用:是一个修饰符,用于修饰成员(成员变量,成员方法) 1.被static修饰后的成员变量只有一份 2.当成员 ...

  5. Java中对象初始化过程

    Java为对象初始化提供了多种选项. 当new一个对象的时候,对象初始化开始: 1.首先,JVM加载类(只加载一次,所以,即使多次new对象,下面的代码也只会在第一次new的时候执行一次),此时, 静 ...

  6. OC 构造方法(对象初始化)

    一.构造方法 (一)构造方法的调用 完整的创建一个可用的对象:Person *p=[Person new]; New方法的内部会分别调用两个方法来完成2件事情,1)使用alloc方法来分配存储空间(返 ...

  7. 字符型图片验证码识别完整过程及Python实现

    字符型图片验证码识别完整过程及Python实现 1   摘要 验证码是目前互联网上非常常见也是非常重要的一个事物,充当着很多系统的 防火墙 功能,但是随时OCR技术的发展,验证码暴露出来的安全问题也越 ...

  8. Java 对象初始化

      对象A的创建过程: 1. 构造器实际上是静态方法.当首次创建对象A 或者 A类的静态方法/静态域首次被访问时,Java解释器查找类路径,以定位     A.class文件.(当程序创建第一个对类的 ...

  9. Java对象初始化详解

    在Java中,一个对象在可以被使用之前必须要被正确地初始化,这一点是Java规范规定的.本文试图对Java如何执行对象的初始化做一个详细深入地介绍(与对象初始化相同,类在被加载之后也是需要初始化的,本 ...

随机推荐

  1. lnmp架构实现动态php

    目录 LNMP动态网站php 1.PHP-FastCGI概述 PHP-FPM安装配置 配置PHP与数据库连接 配置PHP新增扩展模块 配置PHP-FPM主要配置 配置PHP-FPM错误日志 1.编译安 ...

  2. iis应用程序池假死问题

     “Comprehensive orientate   16:05:43  查看原文 IIS貌似问题不少 问:IIS 网站 并发连接线不多,但是运行一段时间后 就非常慢,系统资源占用都正常,一回收应用 ...

  3. Robot Framework——百度搜索

     1.创建项目 选择菜单栏file----->new Project 右键点击新建Project,选择new Suite 选项. 右键点击新建测试Suite,选择new Test Case. 完 ...

  4. delphi XE8 NetHTTPRequest NetHTTPClient

    delphi xe8 推出2个新http控件,NetHTTPRequest.NetHTTPClient 可以调用ASP.Net 一般应用程序获取网页数据,用旧的控件idhttp控件也可以,推荐用新的这 ...

  5. git_基本使用

    1.默认你已经安装了,git的客户端,这里我们使用git bash操作. 2.执行git init命令:     git ini 3.在本地创建ssh key: ssh-keygen -t rsa - ...

  6. TCP/IP知识总结(TCP/IP协议族读书笔记一)

    一.简述TCP/IP协议 Transmission Control Protocol/Internet Protocol的简写,即传输控制协议/互联网互联协议,又名网络通信协议.是Internet最基 ...

  7. consul service

    {      "name": "consul-agent (host:{{ .MONITOR_CONSUL }})",      "command&q ...

  8. VUE递归树形目录(vue递归组件)的使用

    1.html <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" ...

  9. R学习笔记-安装R和RStudio,注意RStudio的版本需要与操作系统版本匹配

    1.安装步骤:先安装R,再安装RStudio RStudio是R的集成开发工具,本身不带R环境. 2.从当前R的官网和RStudio下载的R和RStudio的版本分别为: A .For Windows ...

  10. memcache can't run as root without the -u switch

    memcached是一款高速.分布式的内存缓存系统.其官方主页在http://www.danga.com/memcached/1.安装前的准备要安装memcached,需要有libevent的支持.c ...