53-54 - 被遗弃的多重继承
创始人
2024-02-26 18:13:53
0

---- 整理自狄泰软件唐佐林老师课程

1. 问题

C++是否允许一个类继承自多个父类?

1.1 C++中的 多重继承

  • C++支持编写多重继承的代码
    • 一个子类可以拥有多个父类
    • 子类拥有所有父类的成员变量
    • 子类继承所有父类的成员函数
    • 子类对象可以当作任意父类对象使用

1.2 多重继承的语法规则

在这里插入图片描述

  • 多重继承的本质单继承相同

2. 多重继承问题一

在这里插入图片描述

  • 编程实验:多重继承问题一
#include 
#include using namespace std;class BaseA
{int ma;
public:BaseA(int a){ma = a;}int getA(){return ma;}
};class BaseB
{int mb;
public:BaseB(int b){mb = b;}int getB(){return mb;}
};class Derived : public BaseA, public BaseB
{int mc;
public:Derived(int a, int b, int c) : BaseA(a), BaseB(b){mc = c;}int getC(){return mc;}void print(){cout << "ma = " << getA() << ", "<< "mb = " << getB() << ", "<< "mc = " << mc << endl;}
};int main()
{cout << "sizeof(Derived) = " << sizeof(Derived) << endl;    // 12Derived d(1, 2, 3);d.print();cout << "d.getA() = " << d.getA() << endl;cout << "d.getB() = " << d.getB() << endl;cout << "d.getC() = " << d.getC() << endl;cout << endl;BaseA* pa = &d;BaseB* pb = &d;cout << "pa->getA() = " << pa->getA() << endl;cout << "pb->getB() = " << pb->getB() << endl;cout << endl;void* paa = pa;void* pbb = pb;if( paa == pbb ){cout << "Pointer to the same object!" << endl; }else{cout << "Error" << endl;}cout << "pa = " << pa << endl;cout << "pb = " << pb << endl;cout << "paa = " << paa << endl;cout << "pbb = " << pbb << endl; return 0;
}

在这里插入图片描述
在这里插入图片描述

  • 通过 多重继承 得到的对象可能拥有 “不同的地址

    • 解决方案:无

在这里插入图片描述

3. 多重继承问题二

  • 多重继承可能产生冗余的成员

在这里插入图片描述
Doctor中有两个m_name、m_age

  • 编程实验:多重继承问题二
#include 
#include using namespace std;class People
{string m_name;int m_age;
public:People(string name, int age){m_name = name;m_age = age;}void print(){cout << "Name = " << m_name << ", "<< "Age = " << m_age << endl;}
};class Teacher : virtual public People
{
public:Teacher(string name, int age) : People(name, age){}
};class Student : virtual public People
{
public:Student(string name, int age) : People(name, age){}
};class Doctor : public Teacher, public Student
{
public:Doctor(string name, int age) : Teacher(name, age), Student(name, age), People(name, age){}
};int main()
{Doctor d("Delphi", 33);d.print();return 0;
}

在这里插入图片描述

  • 当多重继承关系出现闭合时将产生数据冗余的问题

    • 解决方案:虚继承
      在这里插入图片描述
  • 虚继承能够解决 数据冗余 问题

  • 中间层父类 不再关心顶层父类的初始化(中间层父类不调用父类构造函数)

  • 最终子类 必须直接调用 顶层父类的构造函数

问题:当架构设计中需要继承时,无法确定使用直接继承还是虚继承

4. 多重继承问题三

  • 多重继承可能产生 多个虚函数表
    在这里插入图片描述

  • 编程实验:多重继承问题三

#include 
#include using namespace std;class BaseA
{
public:virtual void funcA(){cout << "BaseA::funcA()" << endl;}
};class BaseB
{
public:virtual void funcB(){cout << "BaseB::funcB()" << endl;}
};class Derived : public BaseA, public BaseB
{};int main()
{Derived d;BaseA* pa = &d;BaseB* pb = &d;BaseB* pbe = (BaseB*)pa;    // oops!!BaseB* pbc = dynamic_cast(pa);cout << "sizeof(d) = " << sizeof(d) << endl;cout << "Using pa to call funcA()..." << endl;pa->funcA();cout << "Using pb to call funcB()..." << endl;pb->funcB();cout << "Using pbc to call funcB()..." << endl;pbc->funcB();cout << endl;cout << "pa = " << pa << endl;cout << "pb = " << pb << endl;cout << "pbe = " << pbe << endl;cout << "pbc = " << pbc << endl;return 0;
}

在这里插入图片描述

需要进行强制类型转换时,C++中推荐使用新式类型转换关键字
解决方案:dynamic_cast
在这里插入图片描述
在这里插入图片描述

5. 正确的使用多重继承

  • 工程开发中的“多重继承”方式:
    • 单继承某个类 + 实现(多个)接口

在这里插入图片描述

  • 编程实验:正确的多继承方式
#include 
#include using namespace std;class Base
{
protected:int mi;
public:Base(int i){mi = i;}int getI(){return mi;}bool equal(Base* obj){return (this == obj);}
};class Interface1
{
public:virtual void add(int i) = 0;virtual void minus(int i) = 0;
};class Interface2
{
public:virtual void multiply(int i) = 0;virtual void divide(int i) = 0;
};class Derived : public Base, public Interface1, public Interface2
{
public:Derived(int i) : Base(i){}void add(int i){mi += i;}void minus(int i){mi -= i;}void multiply(int i){mi *= i;}void divide(int i){if( i != 0 ){mi /= i;}}
};int main()
{Derived d(100);Derived* p = &d;Interface1* pInt1 = &d;Interface2* pInt2 = &d;cout << "p->getI() = " << p->getI() << endl;    // 100pInt1->add(10);pInt2->divide(11);pInt1->minus(5);pInt2->multiply(8);cout << "p->getI() = " << p->getI() << endl;    // 40cout << endl;cout << "pInt1 == p : " << p->equal(dynamic_cast(pInt1)) << endl;cout << "pInt2 == p : " << p->equal(dynamic_cast(pInt2)) << endl;return 0;
}

在这里插入图片描述
在这里插入图片描述

  • 一些有用的工程建议
    • 先继承自一个父类,然后实现多个接口
    • 父类中提供equal()成员函数
    • equal()成员函数用于判断指针是否指向当前对象
    • 与多重继承相关的强制类型转换使用dynamic_cast完成

6. 小结

  • C++支持多重继承的编程方式
  • 多重继承容易带来问题:
    • 可能出现“同一个对象的地址不同”的情况
    • 虚继承可以解决数据冗余的问题
    • 虚继承使得架构设计可能出现问题
  • 多重继承中可能出现多个虚函数表指针VPTR
  • 与多重继承相关的强制类型转换用dynamic_cast完成
  • 工程开发中采用“单继承多接口”的方式使用多继承
  • 父类提供成员函数用于判断是否指向当前对象

相关内容

热门资讯

湖北一男子当街拦车砸玻璃,警方... 来源:新京报 编辑:王伟 版权归原作者所有 如有侵权请及时联系
员工偷拍单位需要担责吗?被偷拍... 近日,互联网上出现“天津一医院女护士被偷拍”的视频,引发社会关注。12月19日,天津市公安局西青分局...
俄大使:俄罗斯不再将伯尔尼视为... 【俄罗斯驻瑞士大使谢尔盖·加尔莫宁向卫星通讯社表示,瑞士完全采取亲乌克兰的立场,俄罗斯不再将伯尔尼视...
男子当街拦车砸玻璃,一女童站在... 12月20日,湖北大冶市网民发视频称,一名男子在新冶大道附近当街拦车砸玻璃,期间车外有女童哭喊。 ...
知名短剧女演员承认对助理动手:... 近日,知名短剧女演员被曝和助理发生肢体冲突,双方冲突录音曝光。两人疑因工资结算起争执。 12月21日...
5915万元索赔落空!德龙汇能... 深圳商报·读创客户端记者 穆砚 12月19日晚间,德龙汇能集团股份有限公司(以下简称“德龙汇能”或“...
怎样找到高效刑事律师,赵可律师... 影响刑事律师效率的因素在寻找高效刑事律师时,需考虑多方面因素。 专业经验是关键,经验丰富的律师见过各...
《藏在故宫里的法律印迹》讲座在... 讲座现场 日前,复旦大学法学院优秀校友、故宫博物院研究馆员张剑虹研究员以“藏在故宫里的法律印迹”为题...