多重继承引起的二义性问题和虚基类

多重继承容易引起的问题就是因为继承的成员同名而产生的二义性问题。

例:类A和类B中都有成员函数display和数据成员a,类C是类A和类B的直接派生类

情况一:

cpp 复制代码
class A
{
    public:
        int a;
        void display();
};
class B
{
    public:
        int a;
        void display;
};
class C:public A,public B//公用继承
{
    public:
        int b;
        void show();
};

如果在main函数中定义类C的对象c1并调用数据成员a和成员函数display

cpp 复制代码
C c1;
c1.a=3;//引用c1的数据成员a
c1.dispaly();//调用c1的成员函数display

由于基类A和基类B都有数据成员a和成员函数display,编译系统无法判别要访问的是哪一个基类的成员,因此,程序编译出错。

解决方法1、用基类名来限定

cpp 复制代码
c1.A::a=3;//引用c1对象中的基类A的数据成员a
c2.A::display();//引用c1的成员函数display

若派生类C中的成员函数show访问基类A的display和a,可以不必写对象名而直接写

cpp 复制代码
c1.A::A=3;//引用c1对象中的基类A的数据成员a
c1.A::display();//调用c1对象中的基类A的成员函数display

不同的成员函数,只有在函数名和参数个数相同、类型相匹配的情况下才发生同名覆盖,如果只有函数名相同而参数不同,不会发生同名覆盖,而属于函数重载。

虚基类的作用:

C++提供虚基类的方法,使得在继承间接共同基类时只保留一份成员。

使用虚基类发方法:

class A//声明基类A

{.........};

class B:virtual public A//声明类B是类A的公用派生类,A是B的虚基类

{......};

class C:virtual public A//声明类C是类A的公用派生类,A是C的虚基类

{......};

注意:++虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。++

声明虚基类的一般形式为:

class 派生类名:virtual 继承方式 基类名

经过上面的声明后,当基类通过多条派生路径被一个派生类继承时,该派生类只继承该基类一次,也就是说,基类成员只保留一次。

二、虚基类的初始化

class A//定义基类A

{

A(int i){}//基类构造函数,有一个参数

.......};

class B:virtual public A//A作为B的虚基类

{

B(int n):A(n){}//类B构造函数,在初始化表中对虚基类初始化

...};

class C:virtual public A//A作为C的虚基类

{

C(int n):A(n){}

......};//类C的构造函数,在初始化表中对虚基类初始化

class D:public B,public C//类D的构造函数,在初始化表中对所有基类初始化

{

D(int n):A(n),B(n),C(n){}

......};

应用:

cpp 复制代码
#include<iostream>
#include<string>
using namespace std;
class Person
{
public:
	Person(string nam, char s, int a) {
		name = nam;
		sex = s;
		age = a;
	}
protected:
	string name;
	char sex;
	int age;
};
class Teacher :virtual public Person
{
public:
	Teacher(string nam, char s, int a, string t) :Person(nam, s, a)
	{
		title = t;
	}
protected:
	string title;
};
class Student :virtual public Person
{
public:
	Student(string nam, char s, int a, float sco) :Person(nam, s, a)
	{
		score = sco;
	}
protected:
	float score;
};
class Graduate :virtual public Teacher, public Student//Teacher和Student为直接基类
{
public:
	Graduate(string nam, char s, int a, string t, float sco, float w) :Person(nam, s, a), Teacher(nam, s, a, t), Student(nam, s, a, sco)
	{
		wage = w;
	}
	void show()
	{
		cout << "name:" << name << endl;
		cout << "age:" << age << endl;
		cout << "sex:" << sex << endl;
		cout << "title:" << title << endl;
		cout << "score:" << score << endl;
		cout << "wage:<" << wage << endl;
	}
private:
	float wage;
};
int main()
{
	Graduate grad1("yan_tong", 'f', 24, "saadistant", 99.9, 3000);
	grad1.show();

	return 0;
}

运行结果:

相关推荐
肉肉不吃 肉1 分钟前
ES6 模块化 与 CommonJS 的核心概念解析
开发语言·javascript·es6
明月醉窗台9 分钟前
Qt 入门 6 之布局管理
c语言·开发语言·c++·qt
海底火旺13 分钟前
寻找缺失的最小正整数:从暴力到最优的算法演进
javascript·算法·面试
碎梦归途13 分钟前
23种设计模式-结构型模式之适配器模式(Java版本)
java·开发语言·jvm·单例模式·设计模式·适配器模式
优美的赫蒂16 分钟前
理解欧拉公式
线性代数·算法·数学建模
云小逸25 分钟前
【C++】继承
开发语言·c++
Pasregret30 分钟前
策略模式:动态切换算法的设计智慧
算法·bash·策略模式
风静雪冷33 分钟前
Ubuntu中选择Python虚拟环境
开发语言·python
努力学习的小廉1 小时前
【C++】 —— 笔试刷题day_21
开发语言·c++·算法
YuforiaCode1 小时前
第十四届蓝桥杯 2023 C/C++组 冶炼金属
c语言·c++·蓝桥杯