zoukankan      html  css  js  c++  java
  • C++ 派生类构造函数和析构函数

    几个问题

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

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

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

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

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

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

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


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

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

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

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

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

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

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

    class BC
    {
    public:
        BC( )
        {
            x = y = -1;
        }
    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 =100)
        {
            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=0; i<si; i++)
                names[i] = s[i];
                //派生类构造函数执行前,基类构造必须完成。
    
        }
    };

    执行构造函数的顺序:

    1. 基类的构造函数

    2. 子对象的构造函数

    3. 其他数据成员初始化

    继承下的析构函数 Destructors Under Inheritance

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

  • 相关阅读:
    数据库Tsql语句创建--约束--插入数据
    数据绑定的知识点<%%>,<%#%>,<%=%>
    一般处理程序cookie和session+末尾的多选框,下拉框
    拼接字符串
    李航统计学习方法(第二版)(十四):线性支持向量机与软间隔最大化
    李航统计学习方法(第二版)(十三):线性可分支持向量机与硬间隔最大化
    python 并发专题(十二):基础部分补充(四)协程
    python 并发专题(十一):基础部分补充(三)线程
    python 并发专题(十):基础部分补充(二)线程
    python 并发专题(九):基础部分补充(一)进程
  • 原文地址:https://www.cnblogs.com/wkfvawl/p/10738409.html
Copyright © 2011-2022 走看看