当前位置: 首页 > 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都不是对方的基类。(虚二义性与访问规则无关)

相关文章:

  • 一些git的常见操作记录
  • 2556. 第八大奇迹
  • 机器学习-朴素贝叶斯分类
  • IP协议解析
  • RabbitMQ 集群与高可用方案设计(二)
  • 对数的运算困惑
  • C# Windows Forms应用程序-003
  • 数据洪流下,如何让数据库管理不再成为效率瓶颈?
  • 总结:进程和线程的联系和区别
  • 记录一个有用的tcpdump命令
  • Filter和Interceptor详解(一文了解执行阶段及其流程)
  • C#面试问题2
  • QT中常用的类
  • Cadence学习笔记之---PCB过孔替换、封装更新,DRC检查和状态查看
  • 三极管射极跟随器(推挽放大电路)
  • 大语言模型 提示词的少样本案例的 演示选择与排序新突破
  • 单片机上按键功能通常都是用什么方法写?
  • 协同过滤实现电影推荐
  • Java中Map集合的遍历方式详解
  • 【密码学——基础理论与应用】李子臣编著 第十三章 数字签名 课后习题
  • 北京网站建设哪个好/2024年瘟疫大爆发
  • 石家庄电子商务网站建设/宁德市区哪里好玩
  • 书籍扉页页面设计模板/seo流量的提升的软件
  • 即墨网站建设公司/武汉网络推广平台
  • 网站qq交谈怎么做的/合肥seo
  • 做最好的言情网站/军事新闻最新24小时