C++多继承

在前面所有的例子中,派生类都只有一个基类,我们称这种情况为单继承。而在 C++ 中,一个派生类中允许有两个及以上的基类,我们称这种情况为多继承

单继承中,派生类是对基类的特例化,例如编程类书籍是书籍中的特例。而多继承中,派生类是所有基类的一种组合。

【例 1】
class teacher
{
public:
    void settitle(char *a){title = a;}
    char * gettitle(){return title;}
private:
    char * title; //职称
};

class cadre
{
public:
    void setpost(char *a){post = a;}
    char * getpost(){return post;}
private:
    char *post; //职务
};

class teacher_cadre: public cadre, public teacher
{
public:
    void setwages(int a){wages = a;}
    int getwages(){return wages;}
private:
    int wages; //工资
};
在本例中我们定义了三个类,一个是教师类,该类中有一个成员变量职称 title;另一个类是干部类,这个类中有一个成员变量职务 post;最后定义了一个类 teacher_cadre,这个类是 cadre 和 teacher 类的派生类,它也有一个新增的成员变量 wages,表示工资。很明显,教职工干部 teacher_cadre 类是 cadre 类和 teacher 类的组合。

在多继承中,派生类继承了所有基类中的所有成员变量和成员函数,这些继承过来的成员变量及成员函数其访问规则与单继承是相同的,下面我们将例 1 中的 teacher_cadre 类中的所有成员及访问属性列于下表中。

成员 属性 来源
title 不可访问 teacher 类
settitle public teacher 类
gettitle public teacher 类
post 不可访问 cadre 类
setpost public cadre 类
getpost public cadre 类
wages private teacher_cadre 类新增
setwages public teacher_cadre 类新增
getwages public teacher_cadre 类新增

使用多继承可以描述事物之间的组合关系,但是如此一来也可能会增加命名冲突的可能性,冲突可能很有可能发生在基类与基类之间,基类与派生类之间。命名冲突是必须要解决的问题。

【例 2】
#include <iostream>
using namespace std;

class A
{
public:
    void setx(int a){x = a;}
    int getx(){return x;}
private:
    int x;
};

class B
{
public:
    void setx(int a){x = a;}
    int getx(){return x;}
private:
    int x;
};

class C: public A, public B
{
public:
    void setx(int a){x = a;}
    int getx(){return x;}
private:
    int x;
};

int main()
{
    C test;
    test.setx(10);
    test.B::setx(20);
    test.A::setx(30);
    return 0;
}

这个例子是一个非常极端的例子,只是为了说明命名冲突问题。有三个类 A、B 和 C,其中 C 类继承自类 A 和类 B。在三个类中我们都有一个成员变量,变量名恰好都为 x,然后成员函数都名为 setx() 和 getx()。由于两个基类和派生类中出现了命名冲突,因此产生了遮蔽的情况。为了解决命名冲突问题我们只能采用域解析操作符来区分具体所调用的类中的成员函数。