几个问题

一个类的各数据成员的构造顺序?

按他们在类定义中出现的先后顺序:先定义者先构造。

类的对象成员的构造函数与类自身的构造函数的执行顺序?

先执行对象成员的构造函数,再执行类自身的构造函数。

构造顺序与析构顺序的关系?
二者顺序相反:先构造者,后析构。

构造函数和析构函数用来创建和释放该类的对象,当这个类是派生类时,其对象的创建和释放应与其基类对象及成员对象相联系。
在声明派生类时,一般还应当自己定义派生类的构造函数和析构函数,因为构造函数和析构函数是不能从基类继承的

派生类对象的创建和初始化与基类对象的创建和初始化有关。即构造派生类对象时,要对其基类数据成员、所含对象成员的数据成员以及其他的新增数据成员一起进行初始化。这种初始化工作是由派生类的构造函数来完成的。
派生类成员包括两部分:
(1)从基类继承的成员:由基类构造函数完成
(2)自身定义的成员: 由派生类构造函数完成

在派生类中,构造基类数据成员的可能方式:

方式一,在派生类中直接对基类型数据成员初始化:

class BC
{
public:
BC( )
{
x = y = -;
}
private:
int x, y;
};
class DC : public BC
{
public:
DC( )
{
x = y = -;//错误,不能构造基类的私有成员  error C2248: “B::x”: 无法访问 private 成员(在“B”类中声明)
}
private:
string S;
};

方式二,显示调用基类构造函数

class BC
{
public:
BC( )
{
x = y = -;
}
private:
int x, y;
};
class DC : public BC
{
public:
DC( )
{
BC( );//这是构造后才调用,语义错误
}
private:
string S;
};

这样的程序可以编译通过,但语意错误,这是派生类先构造后,在调用基类的构成函数。

正确构成基类数据成员的方式为:

class BC
{
public:
BC( )
{
x = y = -;
}
private:
int x, y;
};
class DC : public BC
{
public:
DC( ) : BC( ), S("派生类"), { }//初始化列表
private:
string S;
};

在创建派生类的对象时,需要调用基类的构造函数:初始化派生类对象从基类继承的成员。在执行一个派生类的构造函数之前,总是先执行基类的构造函数。

调用基类构造函数的两种方式:
(1)显式方式:在派生类的构造函数中,通过参数化表为基类的构造函数提供参数
        derived::derived(arg_derived-list):base(arg_base-list)
(2)隐式方式:在派生类/基类的构造函数都缺省时,派生类的构造函数则自动调用基类的默认构造函数。

在一个多层次的继承层次结构中,一个派生类对象的创建时,其构造函数的调用有点类似于多米诺骨牌效应 (domino effect)

列出了不同情况下的派生类构造函数要求:

上面的例子其实也很好理解,我们知道:在执行一个派生类的构造函数之前,总是先执行基类的构造函数。

1.如果基类中无构造函数,那么对于派生类来说,不管派生类是何种构造函数,编译器都会先调用的执行基类的缺省构造函数,然后再执行派生类的构造函数。

2.如果基类是一个无缺省参数的构造函数,那么对于派生类一旦没有构造函数,那么就不会自动的先构造基类的构造函数,这是不允许的。

3.如果基类中有缺省参数的构造函数B(),那么派生类中没有构造函数也是允许的,编译器会自动调用。

通常, 一个基类有一个缺省构造函数。

以下做法是有其实际意义的:当一个派生类对象被创建时会引起某个基类的构造函数的执行。
 (这条建议在派生类新增成员依赖于基类成员时体现非常明显)

class Team
{
public:
Team(int len =)
{
names = new string[maxno = len ];
//基类构造函数完成其成员初始化,供派生类构造使用。
}
protected:
string* names;
int maxno;
}; class BaseballTeam : public Team
{
public:
BaseballTeam(const string s[], int si)
: Team(si)//为支持派生类构造本意,必须明确调用基类构造函数。 {
for(int i=; i<si; i++)
names[i] = s[i];
//派生类构造函数执行前,基类构造必须完成。 }
};

执行构造函数的顺序:

1. 基类的构造函数

2. 子对象的构造函数

3. 其他数据成员初始化

继承下的析构函数 Destructors Under Inheritance

class BC
{
public:
BC( )
{
sBC = new char[];
cout << "BC allocates 3 bytes.\n";
}
~BC( )
{
delete [ ] sBC;
cout << "BC free 3 bytes.\n";
}
private:
char* sBC;
};
class DC : public BC
{
public:
DC( )
{
sDC = new char[];
cout << "DC allocates 5 bytes.\n";
}
~DC( )
{
delete [ ] sDC;
cout << "DC free 5 bytes.\n";
}
private:
char* sDC;
};
int main( )
{
DC d;
cout << “-------” << endl;
return ;
}

C++ 派生类构造函数和析构函数的更多相关文章

  1. C# 基类派生类构造函数执行顺序分析

    using System; namespace ConsoleApp1 { class Program{ static void Main(string[] args){ B b = new B(); ...

  2. C++中虚继承派生类构造函数的正确写法

    最近工作中某个软件功能出现了退化,追查下来发现是一个类的成员变量没有被正确的初始化.这个问题与C++存在虚继承的情况下派生类构造函数的写法有关.在此说明一下错误发生的原因,希望对更多的人有帮助. 我们 ...

  3. C++语言笔记系列之十三——派生类构造函数的调用

    1.派生类构造函数的调用 (1)一个基类的全部数据成员均被派生类继承.创建一个派生类对象时.系统在为派生类对象分配单元时一定要为其基类数据成员分配子空间. (2)一个派生类对象在创建时不仅要调用派生类 ...

  4. C++的派生类构造函数是否要带上基类构造函数

    //public:Student(int s_age):People(s_age) //C++的派生类构造函数后面是否带上基类构造函数,取决于基类构造函数是否需要传入参数,如果要参数,就一定带上:不需 ...

  5. C++类构造函数、析构函数运行机理

    http://blog.sina.com.cn/s/blog_6fd68d5f0100n60h.html 前言--构造函数.析构函数的简单理解:1)构造函数---对象被创建时候调用的函数:2)析构函数 ...

  6. c++学习之对象和类——构造函数和析构函数

    再认真理一遍~ 0.类 这里先来定义一个类,便于后面的理解.参考C++ Primer Plus class Stock { private: std::string company; long sha ...

  7. 基础学习:关于this在派生类构造函数中的理解

    https://www.cnblogs.com/Bear-Study-Hard/archive/2006/01/09/313551.html 看了上面这篇文章有感,特做了个小样板,以加深对于this在 ...

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

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

  9. C++构造函数和析构函数顺序

    构造函数    先看看构造函数的调用顺序规则,只要我们在平时编程的时候遵守这种约定,任何关于构造函数的调用问题都能解决:构造函数的调用顺序总是如下:1.基类构造函数.如果有多个基类,则构造函数的调用顺 ...

随机推荐

  1. 一次SQLServer数据库宕机问题

    数据库采用SQL Server 2005版本, 数据库文件约为6G,而LDF日志文件已经高达36G. 服务器开始变的不太稳定 .数据没有成功保存. 打开事件查看器发现很多信息日志 数据库 '' 中的文 ...

  2. 为什么ConcurrentHashMap的读操作不需要加锁?

    我们知道,ConcurrentHashmap(1.8)这个并发集合框架是线程安全的,当你看到源码的get操作时,会发现get操作全程是没有加任何锁的,这也是这篇博文讨论的问题--为什么它不需要加锁呢? ...

  3. ICP点云配准原理及优化

    ICP算法简介 根据点云数据所包含的空间信息,可以直接利用点云数据进行配准.主流算法为最近迭代算法(ICP,Iterative Closest Point),该算法是根据点云数据首先构造局部几何特征, ...

  4. vue问题总结

    1.通过判断绑定class及点击事件总结<a :class='[item.status=="yes" ? `btn-primary` : `btn-danger`]' :ti ...

  5. BZOJ5415:[NOI2018]归程(可持久化并查集,最短路)

    Description Input Output Sample Input1 14 31 2 50 12 3 100 23 4 50 15 0 23 02 14 13 13 2 Sample Outp ...

  6. C#游戏开发中精确的时间调配

    方法一:参考<精通C#游戏编程>一书.根据学习WorldWind源码可知,WorldWind中采用的方法与该方法基本一致. using System; using System.Colle ...

  7. CSS中的before和:after伪元素深入理解

    1.定义: “伪元素”,顾名思义.就是它创建了一个虚假的元素,并且将其虚假的元素插入到目标元素的内容之前或之后. 2:特点: a.它在实际文档中不改变什么,但是对用户可见,可以通过css控制,源码中看 ...

  8. eclipse 格式化快捷键(Ctrl+shift+f)不起作用的解决办法

    eclipse格式化快界面Ctrl+Shift+f不起作用一般是键位冲突所导致的,一般是搜狗输入法的“繁体与简体”中文切换快界面冲突. 把它禁用掉就可以了. 下面是禁用步骤: 点击sougou输入法右 ...

  9. Application Constants

    Application: Application类是Android框架中提供的一个类.本身程序员不需要创建它,只需要继承它既可.并在manifest中进行注册. 它给我们提供了一个一般不会被销毁的全局 ...

  10. E - A Twisty Movement

    A dragon symbolizes wisdom, power and wealth. On Lunar New Year's Day, people model a dragon with ba ...