构造函数的调用顺序是先调用System.Object,再按照层次结构由上向下(基类=》派生类)进行,直到到达编译器要实例化的类为止。在此过程中,每个构造函数都初始化自己类中的字段。编译器先自下而上查找构造函数,然后再自上而下地执行。

以下通过示例说明派生类的构造函数是如何执行的。

1.基类和派生类都未显示定义构造函数

执行GenericCustomer customer = new Nevermore60Customer();语句时,编译器首先找到试图实例化的类的构造函数即Nevermore60Customer 类的默认构造函数,默认Nevermore60Customer 构造函数首先要做的是为其直接基类GenericCustomer运行默认构造函数,然后GenericCustomer构造函数为其直接基类System.Object运行默认构造函数。而System.Object没有任何基类,所以它的默认构造函数直接执行;接着执行GenericCustomer的默认构造函数,将name字段初始化为null;最后执行Nevermore60Customer的默认构造函数,将highCostMinutesUsed字段初始化未0,并退出。

    abstract class GenericCustomer
{
private string name;
} class Nevermore60Customer : GenericCustomer
{
private uint highCostMinutesUsed;
}

2.基类定义了一个无参构造函数

执行GenericCustomer customer = new Nevermore60Customer();语句,构造函数执行顺序:System.Object类的默认构造函数=》GenericCustomer类的显示无参构造函数(将name字段初始化为“<no name>”)=》Nevermore60Customer类的默认无参构造函数(将highCostMinutesUsed字段初始化为0)。

    abstract class GenericCustomer
{
private string name; public GenericCustomer()
:base() //base,调用基类的构造函数(本例中调用System.Object中的构造函数,与默认情况相同,可省略)
{
name = "<no name>";
}
} class Nevermore60Customer : GenericCustomer
{
private uint highCostMinutesUsed;
}

注意:若把GenericCustomer类中的构造函数声明为private,则类Nevermore60Customer会产生一个编译错误。因为编译器试图为Nevermore60Customer类生成默认构造函数时,需要调用类GenericCustomer的无参构造函数,但是这个函数是类GenericCustomer所私有的,其他类无法调用。

3.基类和派生类都定义了有参构造函数

执行GenericCustomer customer = new Nevermore60Customer("LiSi");语句时,构造函数执行顺序:System.Object类的默认构造函数=》GenericCustomer类的有参构造函数(将name字段初始化为“LiSi”)=》Nevermore60Customer类的有参构造函数(什么也不做)。

    abstract class GenericCustomer
{
private string name; public GenericCustomer(string name)
{
this.name = name;
}
} class Nevermore60Customer : GenericCustomer
{
private uint highCostMinutesUsed; public Nevermore60Customer(string name)
:base(name)
{ }
}

注意:若类Nevermore60Customer未定义上面的有参构造函数,则类Nevermore60Customer会产生一个编译错误。因为类Nevermore60Customer生成的默认构造函数会试图调用GenericCustomer类中的无参构造函数,但它并没有这样的函数。

4.派生类中有多个构造函数

执行GenericCustomer customer = new Nevermore60Customer("LiSi");语句时,构造函数执行顺序:System.Object类的默认构造函数=》GenericCustomer类的有参构造函数(将name字段初始化为“LiSi”)=》Nevermore60Customer类中有两个参数的构造函数(将referrerName字段初始化为“None”)=》Nevermore60Customer类中有一个参数的构造函数(什么也不做)

    abstract class GenericCustomer
{
private string name; public GenericCustomer(string name)
{
this.name = name;
}
} class Nevermore60Customer : GenericCustomer
{
private uint highCostMinutesUsed; private string referrerName; public Nevermore60Customer(string name, string referrerName)
: base(name)
{
this.referrerName = referrerName;
} public Nevermore60Customer(string name)
: this(name, "<None>")
{ }
}

参考来源:《C#高级编程(第9版)》

C#派生类的构造函数的更多相关文章

  1. 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)

    [源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...

  2. C++学习17派生类的构造函数

    基类的构造函数不能被继承,在声明派生类时,对继承过来的成员变量的初始化工作也要由派生类的构造函数来完成.所以在设计派生类的构造函数时,不仅要考虑派生类新增的成员变量,还要考虑基类的成员变量,要让它们都 ...

  3. C++:派生类的构造函数和析构函数

    4.2 派生类的构造函数和析构函数4.2.1 派生类构造函数和析构函数的执行顺序 通常情况下,当创建派生类对象时,首先执行基类的构造函数,随后再执行派生类的构造函数:当撤销派生类对象时,则先执行派生类 ...

  4. 【C++学习之路】派生类的构造函数(三)

    三.多层继承的派生类 1.多层继承的派生类只需在构造函数的初始化列表中写出直接基类的构造函数即可 class student { public: student(int n, string nam) ...

  5. 【C++学习之路】派生类的构造函数(一)

    一.简单派生类的构造函数 1.所谓简单派生类,就是指派生类中不包含基类的内嵌对象的派生类. 2.一般来说,这样的派生类的构造函数的形式是: student( int i, string nam, in ...

  6. 【C++继承与派生之二】有子对象的派生类的构造函数

    这是我今天看书刚刚看到的,觉着以前对这一块内容了解不多,所以整理一下分享给大家.首先要介绍一下子对象的概念.类的数据成员不仅可以是int.char这样的基本类型,也可以是类对象,如可以包含这样的数据成 ...

  7. c++, 派生类的构造函数和析构函数 , [ 以及operator=不能被继承 or Not的探讨]

    说明:文章中关于operator=实现的示例,从语法上是对的,但逻辑和习惯上都是错误的. 参见另一篇专门探究operator=的文章:<c++,operator=>http://www.c ...

  8. C++学习之路—继承与派生(二):派生类的构造函数与析构函数

    (根据<C++程序设计>(谭浩强)整理,整理者:华科小涛,@http://www.cnblogs.com/hust-ghtao转载请注明) 由于基类的构造函数和析构函数是不能被继承的,所以 ...

  9. 【C++ Primer 第15章】定义派生类拷贝构造函数、赋值运算符

    学习资料 • 派生类的赋值运算符/赋值构造函数也必须处理它的基类成员的赋值 • C++ 基类构造函数带参数的继承方式及派生类的初始化 定义拷贝构造函数 [注意]对派生类进行拷贝构造时,如果想让基类的成 ...

  10. C++:派生类的构造函数和析构函数的调用顺序

    一.派生类 在C++编程中,我们在编写一个基类的派生类时,大致可以分为四步: • 吸收基类的成员:不论是数据成员还是函数成员,派生类吸收除基类的构造函数和析构函数之外的全部成员. • 改造基类函数:在 ...

随机推荐

  1. 2018-2019-2 20165205 网络对抗技术 Exp9 Web安全基础

    2018-2019-2 20165205 网络对抗技术 Exp9 Web安全基础 1.基础问题 SQL注入攻击原理,如何防御 原理: SQL注入指攻击者在提交查询请求时将SQL语句插入到请求内容中,同 ...

  2. 用Java实现一个二叉树

    介绍 使用Java实现一个int值类型的排序二叉树 二叉树 二叉树是一个递归的数据结构,每个节点最多有两个子节点. 通常二叉树是二分查找树,每个节点它的值大于或者等于在它左子树节点上的值,小于或者等于 ...

  3. C++输入输出流加速器,关闭同步流,ios::sync_with_stdio(false)和 cin.tie(0)

    leetcode练习时,总会发现运行时间短的代码都会有类似: static int x=[](){ std::ios::sync_with_stdio(false); cin.tie(NULL); ; ...

  4. cpu切换线程上下文会耗费多少时间

    cpu切换线程上下文会耗费多少时间,有人在linux下面使用不同的cpu测试过,需要1000ns以上的时间 https://blog.tsunanet.net/2010/11/how-long-doe ...

  5. charles破解https请求

    当你的app包从http升级到https的时候,是不是忽然间发现你的请求抓不到了呢?别担心,只是因为你们的app加密升级了,但是我们还是可以正常破解的.接下来直接谈破解步骤啦: 1:首先打开charl ...

  6. Python程序猿面试杂谈

    不同的互联网公司面试环节不尽相同,总体而言,一般有技术面(1-3轮).leader面(组长/总监/CTO/CEO...).hr面,面试轮数视公司规模和公司实际具体情况而定.下文将就面试的不同环节阐述下 ...

  7. 【AMAD】betamax -- 一个ruby-VCR的模仿品,只支持requests

    简介 动机 作用 用法 个人评分 简介 betamax1会记录你的HTTP操作,可以让你在测试的时候不必重复进行真实的请求. 动机 如果你的代码需要和外部资源一起运作,那么测试这段代码的方法就叫做集成 ...

  8. 应用安全 - 工具|平台 - CDN - 使用|命令 - 汇总

    简介 用途 使用缓存适应高并发请求 功能 ()抗DDOS ()隐藏真实IP 全球DNS地址分布:http://www.ab173.com/dns/dns_world.php全球IP地址段分布:http ...

  9. 蓝鲸 修改主机名重装后初始化不了cmdb安装不了job + 数据采集流程

    1.表象:在部署蓝鲸JOB过程中需要进行RabbitMQ的安装,数据初始化,激活步骤,此问题多发生在此过程 [ root@rbtnodel install)# ./bkcec initdata rab ...

  10. MVC、MVP、MVVM模式的概念与区别

    1. MVC框架 MVC全名是Model View Controller,是模型(model)-视图(view)-控制器(controller)的缩写,一种软件设计典范,用一种业务逻辑.数据.界面显示 ...