子类的构造器及构造原则

父类中如果有标配,重载或默认(需包含默认),子类中可以不用显示的调用父类的构造器
子类只需要对父类负责(初始化父类自己的参数),不需要对父类的父类负责。父类的父类,由父类负责,但是虚继承除外,虚继承中要对虚基类负责
虚基类构造函数的参数必须由最新派生出来的类负责初始化,即使不是直接继承。

#include "stdafx.h"
#include <iostream>
using namespace std;

class A
{
public:
	A()
	{
		cout << "A" << endl;
	}

};


class Father
{
public:
	Father(int i = 0)
	{
		cout << "Father" << endl;
	}

};

class Son :public Father
{
public:
	Son(int j) :Father(j) //父类有默认参数 可以不用显示调用
	{
		cout << "Son" << endl;
	}
};


class GrandSon :public Son
{
public:
	GrandSon(int k) :Son(k)
	{
		cout << "GrandSon" << endl;
	}
	A a; //在本类构造器进入作用域之前调用

};


int _tmain(int argc, _TCHAR* argv[])
{
	Son son(10);

	GrandSon(10); //打断点 可以查看构造的顺序
	//顺序:本类的父类们->类对象数据成员->本类

	return 0;
}

以下转载自:https://blog.csdn.net/maosong00/article/details/12012895

构造方法用来初始化类的对象,与父类的其它成员不同,它不能被子类继承(子类可以继承父类所有的成员变量和成员方法,但不继承父类的构造函数和析构函数)。因此,在创建子类对象时,为了初始化从父类继承来的数据成员,系统需要调用其父类的构造方法。如果没有显式的构造函数,编译器会给一个默认的构造函数,并且该默认的构造函数仅仅在没有显式地声明构造函数情况下创建。

构造原则如下:

    1. 如果子类没有定义构造方法,则调用父类的无参数的构造方法。

    2. 如果子类定义了构造方法,不论是无参数还是带参数,在创建子类的对象的时候,首先执行父类无参数的构造方法,然后执行自己的构造方法。

    3. 在创建子类对象时候,如果子类的构造函数没有显示调用父类的构造函数,则会调用父类的默认无参构造函数。

    4. 在创建子类对象时候,如果子类的构造函数没有显示调用父类的构造函数且父类自己提供了无参构造函数,则会调用父类自己的无参构造函数。

    5. 在创建子类对象时候,如果子类的构造函数没有显示调用父类的构造函数且父类只定义了自己的有参构造函数,则会出错(如果父类只有有参数的构造方法,则子类必须显示调用此带参构造方法)。

    6. 如果子类调用父类带参数的构造方法,需要用初始化父类成员对象的方式;

 记住:子类只能在构造对象时才能默认(或者用初始化列表中显示调用“特定父类构造函数”)调用父类的构造函数,在构造完成后不能像调用父类成员函数一样调用父类构造函数,这样保证了父类构造函数只调用一次的原则;

#include<iostream>
using namespace std;

class Base{ 
public:
    Base(){cout << "none argument !!" << endl;};
    Base(int x1) : x(x1){
    cout<<"Base x1"<<endl;
    }; 
    virtual ~Base(){};

    void func(int x){cout << "Base : " << x << endl;};
private: 
    int x; 
};

class Derive : public Base {
public:
    Derive () : Base(100){};
    virtual ~Derive (){};

//该处覆盖了Base的func方法;

//  void func(int x){cout << "Base : " << x << endl;};         

private:
    int dint;
};

int main()
{ 
    Derive derive;
    Derive::Base(2);      //::表示编译期间确定的调用,不是对象的调用,故可以编译通过;
    // derive.Base();          //该语句会出错,表示不能继承父类的构造函数,对比下一个语句;
    derive.func(2);         //该语句可编译通过,表示能继承父类的普通成员函数。
    Base *pBase = &derive;
    pBase->func(1);
    return 0;
}


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值