对象初始化的完整过程(C#)
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#)的更多相关文章
- JAVA 对象初始化的过程
对象初始化的过程例:Student S = new Student();1.因为new Student()用到了Student类,所以会把它从硬盘上加载进入内存2.如果有static静态代 ...
- Java父类子类的对象初始化过程
摘要 Java基本的对象初始化过程,子类的初始化,以及涉及到父类和子类的转化时可能引起混乱的情况. 1. 基本初始化过程: 对于一个简单类的初始化过程是: static 修饰的模块(static变量和 ...
- Java类与对象初始化的过程(一道经典的面试题)
本文不再以ClassLoader的视角解释这些问题. 首先,Java代码有个特点,就是成员变量可以在前面的方法中使用,在后面定义.这一特性,很多人说Java了不起,可是为什么呢?Java为何能够这样呢 ...
- java基础课程笔记 static 主函数 静态工具类 classpath java文档注释 静态代码块 对象初始化过程 设计模式 继承 子父类中的函数 继承中的构造函数 对象转型 多态 封装 抽象类 final 接口 包 jar包
Static那些事儿 Static关键字 被static修饰的变量成为静态变量(类变量) 作用:是一个修饰符,用于修饰成员(成员变量,成员方法) 1.被static修饰后的成员变量只有一份 2.当成员 ...
- Java中对象初始化过程
Java为对象初始化提供了多种选项. 当new一个对象的时候,对象初始化开始: 1.首先,JVM加载类(只加载一次,所以,即使多次new对象,下面的代码也只会在第一次new的时候执行一次),此时, 静 ...
- OC 构造方法(对象初始化)
一.构造方法 (一)构造方法的调用 完整的创建一个可用的对象:Person *p=[Person new]; New方法的内部会分别调用两个方法来完成2件事情,1)使用alloc方法来分配存储空间(返 ...
- 字符型图片验证码识别完整过程及Python实现
字符型图片验证码识别完整过程及Python实现 1 摘要 验证码是目前互联网上非常常见也是非常重要的一个事物,充当着很多系统的 防火墙 功能,但是随时OCR技术的发展,验证码暴露出来的安全问题也越 ...
- Java 对象初始化
对象A的创建过程: 1. 构造器实际上是静态方法.当首次创建对象A 或者 A类的静态方法/静态域首次被访问时,Java解释器查找类路径,以定位 A.class文件.(当程序创建第一个对类的 ...
- Java对象初始化详解
在Java中,一个对象在可以被使用之前必须要被正确地初始化,这一点是Java规范规定的.本文试图对Java如何执行对象的初始化做一个详细深入地介绍(与对象初始化相同,类在被加载之后也是需要初始化的,本 ...
随机推荐
- 关于&&和||
从alert(1&&2)输出为2谈起 一.先来说说||(逻辑或),从字面上来说,只有前后都是false的时候才返回false,否则返回true. alert(true||false); ...
- canvas填充规则
canvas填充规则 const canvas = document.getElementById('canvas'); const ctx = canvas.getContext('2d'); ct ...
- tensorflow-base_operations
# -*- coding: utf-8 -*-import tensorflow as tf# 基本的常量操作,通过构造函数返回值 定义值的操作operationsa = tf.constant(2) ...
- 二维码名片的格式 - vcard(非常好,可直接添加到手机通讯录)
分享到 一键分享 QQ空间 新浪微博 百度云收藏 人人网 腾讯微博 百度相册 开心网 腾讯朋友 百度贴吧 豆瓣网 搜狐微博 百度新首页 QQ好友 和讯微博 更多... 百度分享 登录|注册 ...
- fireDAC oracle
copy 4 files to D:\oracleapp\Administrator\product\11.2.0\client_1\BIN setup win 64 bit client .down ...
- localStorage,sessionStorage
h5的本地存储. 现代浏览器普遍开始支持H5本地存储,localStorage.sessionStorage.可以用来代替cookie的一部分存储功能,他比cookie存储量更大.比较实用. 两者用法 ...
- Node.js究竟是什么?
来源:https://www.ibm.com/developerworks/cn/opensource/os-nodejs/index.html?ca=drs#ibm-pcon Node 旨在解决 ...
- selenium+java,实现部分截图功能,-针对单个元素的截图
有时候需要元素的截图,不需要整个截图.整理一个针对元素的截图的方法. 创建一个Java类,实现截取元素的方法 package com.lozz.utils; import java.awt.Recta ...
- Maven的pom.xml介绍
6.1 简介 pom.xml文件是Maven进行工作的主要配置文件.在这个文件中我们可以配置Maven项目的groupId.artifactId和version等Maven项目必须的元素:可以配置Ma ...
- vue elementui form表单验证
最近我们公司将前端框架由easyui 改为 vue+elementui .自学vue两周 就开始了爬坑之路.业余时间给大家分享一下心得,技术新手加上第一次分享(小激动),有什么不足的地方欢迎大家指正, ...