当前位置: 首页 > news >正文

C++:多重继承

        MI描述的是有多个直接基类的类。与单继承一样,公有MI表示的也是is-a关系。例如,Worker派生出Singer和Waiter,再从Singer和Waiter派生出SingingWaiter。但多重继承会带来许多问题,最主要的两个便是

  • 从两个不同的基类继承的同名方法
  • 从两个或更多相关基类哪里继承同一个类的多个实例

        就上述两个问题,下面根据举例Worker继承给出相关解决方法

有多少Worker

        Worker派生出Singer和Waiter,Singer和Waiter再派生出SingerWaiter,则SingingWaiter将包含两个Worker,这将可能出现二义性

SingingWaiter ed;
Worker*pw=&ed;//是哪个Worker的地址呢?

        所以这里需要使用类型转换来指定对象。

Worker *pw1=(Waiter*) &ed;//Waiter中的Worker地址
Worker *pw2=(Singer*) &ed;//Singer中的Worker地址

         这使得使用基类指针来引用不同对象变得复杂,但真正的问题是为什么需要多个Worker?SingingWaiter的对象是一个单独的人,不该带有两个Worker对象。为此我们需要引入虚基类来确保从多个类派生出的对象只继承一个基类对象。

虚基类

class Singer:virtual public Worker{...}
class Waiter:public virtual Worker{...}//virtual的位置无关紧要
class SingingWaiter:public Singer,public Waiter{...}

        上述代码声明使得Worker被用作Singer和Waiter的虚基类,且SingingWaiter对象只包含了一个Worker子对象。

        使用虚基类时,需对类构造函数采用一种新的方法。C++在基类是虚的时,禁止信息通过中间类自动传递给虚类。因此派生类实现的多重继承需要直接调用基类的构造函数(与非虚基类不允许越过上一级不同)。

SingingWaiter(const Worker&wk,int p=0,int v=Singer::other):Waiter(wk,p),Singer(wk,v){}
//这里wk信息不会传递到虚基类,因为两条途径不同会产生冲突
//会自动调用Worker的默认构造函数SingingWaiter(const Worker&wk,int p=0,int v=Singer::other):Worker(wk),Waiter(wk,p),Singer(wk,p){}
//这样显式调用构造函数Worker可阻止Worker默认构造函数的调用

        注意:这种用法在虚基类是合法的,且必须这样做;但对于非虚基类,这样是非法的。

使用哪个方法

        我们希望通过SingingWaiter调用show()方法(同时被Singer和Waiter拥有),但直接调用会导致二义性,不知道会调用哪个直接祖先的方法。于是,我们可以通过使用作用域解析运算符来表示意图。

SingingWaiter n("ew",2005,6,soprano)
n.Singer::Show();//调用Singer的版本

        但更好的方法是在SingingWaiter中重新定义Show(),并指出希望使用哪一个Show()。

void SingingWaiter::Show()
{Singer::Show();
}//希望SingingWaiter使用Singer版本的Show()

        在单继承中可以使用递增的方式显示类的信息(基类显示基类的,派生类调用基类方法加上派生类的信息,以此类推)。但多重继承不行,这样会忽略部分信息或是重复显示信息。所以多重继承应使用模块化方法:提供一个Worker组件的方法和一个只显示Waiter组件或Singer组件的方法,再在SingingWaiter方法中调用上述方法。

        另一种方法是将所有数据组件都设置为保护的而非私有的。

虚基类与支配

        使用虚基类将改变C++解析二义性的方式。使用虚基类时,派生类使用从多个基类那里继承得到的同名成员名,如果不使用类名限定,也不一定会导致二义性。这种情况下,如果某个名称优先于其他所有名称,则使用它且不导致二义性。

        如何判断优先性呢?派生类中的名称优先于直接或间接祖先中的相同名称。

class B
{
public:short q();...
};class C:virtual public B
{
public:short q;int omg()...
};
class D:public C
{
...
};
class E:virtual public B
{
private:int omg();...
};class F:public D,public E
{...
};

        类C中q()定义优先于类B中的q()定义,因为C时从B类派生来的。任何一个omg()定义都不优先于其他omg()定义,因为C和E都不是对方的基类。(虚二义性与访问规则无关)

http://www.xdnf.cn/news/659413.html

相关文章:

  • 蓝桥杯b组c++赛道---数位dp
  • git 新建一个分支,怎么首次推到远程仓库
  • 计算机图形学:(四)欧拉角与四元数
  • 尚硅谷redis7 37 redis持久化之AOF简介
  • Unity---OSC(Open Sound Control)、TouchOSC Editor、创建布局
  • Java高频面试之并发编程-21
  • Linux `hostname` 命令深度解析与高阶应用指南
  • Linux中的SELinux
  • RPM之(1)基础使用
  • 【2025】嵌入式软考中级部分试题
  • [特殊字符] useTranslations 客户端使用教程(Next.js + next-intl)
  • n8n中文版安装指南,使用Docker部署N8N中文版
  • 深度学习入门6:pytorch卷积神经网络CNN实现手写数字识别准确率99%
  • 深度学习中的卷积和反卷积
  • 北京大学肖臻老师《区块链技术与应用》公开课:01-课程简介
  • 《软件工程》第 11 章 - 结构化软件开发
  • Qt Creator快捷键合集
  • GESP2024年9月认证C++二级( 第三部分编程题(2)小杨的矩阵)
  • LangChain理解
  • Mybatis框架
  • Redis分布式缓存核心架构全解析:持久化、高可用与分片实战
  • UDP协议原理与Java编程实战:无连接通信的奥秘
  • 【Webtrees 手册】第 4 章 - 编辑指南
  • 通用的管理账号设置设计(一)
  • 02. [Python+Golang+PHP]三数之和,多种语言实现最优解demo
  • 华为OD机试真题——分糖果(2025A卷:100分)Java/python/JavaScript/C/C++/GO最佳实现
  • Linux 网络配置现代实践:Netplan 与 ifcfg 的全景对比与工程指南20250526
  • 身份证二要素核验:数字经济时代的信任基石
  • React从基础入门到高级实战:React 核心技术 - 表单处理与验证深度指南
  • 关于模型记忆力的实现方式