继承和派生

基类和派生类

class Student                             //基类
{
    //...
};

class CollegeStudent : public Student    //大学生是学生的派生类
{
    //...   
};

派生类对于基类继承提供了代码的重用性。派生类的增加部分提供了对原有代码扩充和改进的能力。

继承的特点:

  • 一个派生类可以有一个或多个基类,只有一个基类时叫单继承(每个类可以有多个派生类,但最多有一个基类),有多个基类时叫多继承
  • 继承关系可以是多级的,即类Y继承类X,类Z继承类Y同时存在。
  • 不允许继承循环,即不允许类A继承类B,类B继承类C,类C继承类A同时存在。
  • 基类中能够被继承的部分只能是公有成员和保护成员,私有成员不能被继承

根据继承关键字不同,其继承的方式也不同:

  • public:        表示公有基类               基类的公有成员    =>    派生类的公有成员  ||  基类的保护成员    =>    派生类的保护成员
  • private:      表示私有基类                基类的公有成员和保护成员    =>    派生类的私有成员
  • protected:  表示保护基类                基类的公有成员和保护成员    =>    派生类的保护成员

当基类和派生类中的函数重名时,为了能够在派生类中仍然能调用基类中的函数,可以用 :: 运算符进行调用。

class A
{
public:
    void fun();
};

class B : public A
{
public:
    void fun();
    void g()
    {
        fun();        //类B中的fun()
        A::fun();     //类A中的fun()
    }
};

派生类的构造函数和析构函数

派生类不能继承基类的构造函数和析构函数,但可以在创建派生类对象时,调用基类的构造函数来初始化基类的数据成员。构造函数的调用时先基类后派生类。同样,析构函数的调用顺序与构造函数相反。

派生类构造函数的一般格式为:

派生类名 (派生类参数) : 基类构造函数 (参数表) ,子对象名 (参数表)
{
    派生类数据成员初始化;
}

派生类构造函数的调用顺序:(先基类后派生类

  • 基类的构造函数是按继承声明的顺序进行调用的。
  • 调用子对象构造函数时,是按着在类中声明的顺序进行调用的。
  • 派生类构造函数体内的语句也是按着声明时的顺序执行的。

派生类在创建对象时,系统将自动调用其构造函数,以及其基类的构造函数。如果派生类的基类无构造函数或构造函数无参数,则派生类可以定义或不定义构造函数。如果派生类的基类有构造函数,并且构造函数有参数,则派生类必须定义构造函数,并且要通过定义的构造函数将参数传递给基类的构造函数

class A
{
    int num;
    float grade;
public:
    A(int n,float v) : num(n),grade(v)           //带参数构造函数
    {    cout<<"the A initialized"<<endl;    }
};
class B : public A                 //在调用B的对象时,从基类开始初始化,因此先调用基类构造函数
{
    int num;
    float grade;
public:
    B(void) : A(10000,0)                         //无参数构造函数,要为基类的构造函数设置参数
    {    cout<<"the B initialized"<<endl;    }
    B(int n, float x, double dx) : A(n, x)       //带参数构造函数,要为基类的构造函数设置参数
    {    cout<<"the B initialized"<<endl;    }
};

继承的二义性

class A
{
public:
    void f();
};

class B
{
public:
    void g();
    void f();
};
class C :public A,public B
{
public:
    void h();
    f();
};

在这里调用函数 f() 就产生了二义性,因为不能明确调用的是类A中的 f() 还是类B中的,我们可以通过在类C中重新定义一个新的 f(),即 void f(); 的方式消除二义性。函数的实现如下:

void C::f()
{
    A::f();
}

虚基类(virtual)

如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的名字进行访问时,可能产生二义性。因此我们使用虚基类来使公共基类在派生类中只产生一个子对象,此时不管从哪条路径去访问公共基类的子对象时,都是访问的一个子对象,这样就不会产生二义性。

class B
{
public:
    int b;
};

class B1 : virtual public B
{
public:
    int b1;
};

class B2 : virtual public B
{
public:
    int b2;
};

class C : public B1,public B2
{
private:
    int c;
};

void main()
{
    C cc;
    cc.b;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

肥羊汤

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值