本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数

在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数。下面就是一个例子:

  1. class CBaseA
  2. {
  3. public:
  4. virtual void TestA();
  5. };
  6. class CBaseB
  7. {
  8. public:
  9. virtual void TestB();
  10. };
  11. class CDerived : public CBaseA, public CBaseB
  12. {
  13. public:
  14. virtual void TestA(); // 重写基类CBaseA中的虚函数TestA()
  15. virtual void TestB(); // 重写基类CBaseB中的虚函数TestB()
  16. };
  17. void Test()
  18. {
  19. CDerived D;
  20. CBaseA *pA = &D;
  21. CBaseB *pB = &D;
  22. pA->TestA(); // 调用类CDerived的TestA()函数
  23. pB->TestB(); // 调用类CDerived的TestB()函数
  24. }

可是,如果两个基类中有一个相同原型的虚函数,例如下面这样:

  1. class CBaseA
  2. {
  3. public:
  4. virtual void Test();
  5. };
  6. class CBaseB
  7. {
  8. public:
  9. virtual void Test();
  10. };

怎样在派生类中重写这两个相同原型的虚函数呢?
    也许这种情况并不常见,可是这种情况却确实存在。比如说开发的时候使用的两个类库是不同的厂商提供的,或者说这两个类库是由公司的不同开发小组开发的。对前者来说,修改基类的接口是不可能的;对后者来说,修改接口的代价很大。
    如果在派生类中直接重写这个虚函数,那么2个基类的Test()虚函数都将被覆盖。这样的话就只能有一个Test()的实现,而不是像前面的例子那样有不同的实现。

  1. class CDerived : public CBaseA, public CBaseB
  2. {
  3. public:
  4. virtual void Test();
  5. };
  6. void Test()
  7. {
  8. CDerived D;
  9. CBaseA *pA = &D;
  10. CBaseB *pB = &D;
  11. // 下面2行代码都将调用类CDerived的Test()函数
  12. pA->Test();
  13. pB->Test();
  14. }

为了实现第一个例子中的那样,在派生类CDerived中重写不同基类中相同原型的虚函数Test(),可以使用下面的方法。
    首先,不需要对2个基类进行任何修改(在实际的开发当中,修改基类的可能性非常小)。

  1. class CBaseA
  2. {
  3. public:
  4. virtual void Test();
  5. };
  6. class CBaseB
  7. {
  8. public:
  9. virtual void Test();
  10. };

现在,为这个继承体系添加2个中间类,分别从2个基类派生。

  1. class CMiddleBaseA : public CBaseA
  2. {
  3. private:
  4. // 真正的实现函数
  5. // 设置为纯虚函数,在派生类里必须实现
  6. virtual void CBaseA_Test() = 0;
  7. // 改写继承下来的虚函数
  8. // 仅仅直接调用真正的实现函数
  9. virtual void Test()
  10. {
  11. CBaseA_Test();
  12. }
  13. };
  14. // 与类CMiddleBaseA采用相同的方法
  15. class CMiddleBaseB : public CBaseB
  16. {
  17. private:
  18. virtual void CBaseB_Test() = 0;
  19. virtual void Test()
  20. {
  21. CBaseB_Test();
  22. }
  23. };

然后,类CDerived以上面2个中间类作为基类来派生。分别重写上面2个基类中原型不同的纯虚函数,添加不同的实现代码。

  1. class CDerived : public CMiddleBaseA, public CMiddleBaseB
  2. {
  3. private:
  4. // 重写从中间类继承下来的虚函数
  5. virtual void CBaseA_Test(); // 这里实际上是重写CBaseA的Test()
  6. virtual void CBaseB_Test(); // 这里实际上是重写CBaseB的Test()
  7. };
  8. void Test()
  9. {
  10. CDerived D;
  11. CBaseA *pA = &D;
  12. CBaseB *pB = &D;
  13. // 调用类CBaseA的Test()函数
  14. // 由于C++多态的特性,实际上调用的是类CDervied中的CBaseA_Test()函数
  15. pA->Test();
  16. // 调用类CBaseB的Test()函数
  17. // 由于C++多态的特性,实际上调用的是类CDervied中的CBaseB_Test()函数
  18. pB->Test();
  19. }

现在以上面代码中的pA->Test();这行代码来说明上面的方案是怎么实现的。
    首先,由于虚函数Test()在类CBaseA的派生类CMiddleBaseA中被重写,所以这行代码会去调用类CMiddleBaseA的Test()函数;
    然后,类CMiddleBaseA的Test()函数会去调用实现函数CBaseA_Test();
    最后,由于虚函数CBaseA_Test()在类CMiddleBaseA的派生类CDerived中被重写,所以真正调用的是类CDerived中的CBaseA_Test()函数。
    同样的道理,代码pB->Test();实际上调用的是类CDervied中的CBaseB_Test()函数。

通过上面的方法就可以在C++多继承中重写不同基类中相同原型的虚函数。

【转载】 C++多继承中重写不同基类中相同原型的虚函数的更多相关文章

  1. C++基类的析构函数定义为虚函数的原因

    1:每个析构函数只会清理自己的成员(成员函数前没有virtual).2:可能是基类的指针指向派生类的对象,当析构一个指向派生类的成员的基类指针,这时程序不知道这么办,可能会造成内存的泄露,因此此时基类 ...

  2. cc31a_demo--CppPrimer_静态成员与继承-在派生类中访问基类中的static成员的方法

    //*基类中的static成员,在整个继承层次中只有一个实例 //*在派生类中访问基类中的static成员的方法 //1.基类名::成员名 //2.子类名::成员名 //3.对象.成员名 //4.指针 ...

  3. 面向对象编程(九)——面向对象三大特性之继承以及重写、Object类的介绍

    面向对象三大特性 面向对象三大特征:继承 :封装/隐藏 :多态(为了适应需求的多种变化,使代码变得更加通用!) 封装:主要实现了隐藏细节,对用户提供访问接口,无需关心方法的具体实现. 继承:很好的实现 ...

  4. Filter组件开发中的SDK基类分析

    DirectShow SDK提供了一套开发Filter的基类源代码.基于这些基类开发Filter将大大简化开发过程. 1.CBaseObject 大部分SDK类都从CBaseObject类(参见com ...

  5. 自定义继承于Page的基类

    自定义继承于Page的基类:MyBasePage[校验用户是否登录,如果登录则获取用户信息,否则跳转到登录页面]============================================ ...

  6. 基类中定义的虚函数在派生类中重新定义时,其函数原型,包括返回类型、函数名、参数个数、参数类型及参数的先后顺序,都必须与基类中的原型完全相同 but------> 可以返回派生类对象的引用或指针

      您查询的关键词是:c++primer习题15.25 以下是该网页在北京时间 2016年07月15日 02:57:08 的快照: 如果打开速度慢,可以尝试快速版:如果想更新或删除快照,可以投诉快照. ...

  7. 读书笔记 effective c++ Item 43 了解如何访问模板化基类中的名字

    1. 问题的引入——派生类不会发现模板基类中的名字 假设我们需要写一个应用,使用它可以为不同的公司发送消息.消息可以以加密或者明文(未加密)的方式被发送.如果在编译阶段我们有足够的信息来确定哪个信息会 ...

  8. C++可继承的单例基类模板

    目录 一.介绍 二.代码 三.关键处 五.参考资料 一.介绍 最近在写一个项目,其中用到好几个单例,类本身的设计不是很复杂,但是如果每个都写一遍单例又觉得有点冗余:所以查资料写了一个单例基类模板,只要 ...

  9. why pure virtual function has definition 为什么可以在基类中实现纯虚函数

    看了会音频,无意搜到一个frameworks/base/include/utils/Flattenable.h : virtual ~Flattenable() = 0; 所以查了下“纯虚函数定义实现 ...

随机推荐

  1. Keepalived+Nginx架构整理版

    Keepalived介绍 keepalived是一个类似于layer3, 4, 5 交换机制的软件,也就是我们平时说的第3层.第4层和第5层交换.Keepalived的作用是检测web服务器的状态,如 ...

  2. (转)php-curl响应慢(开发微信授权登陆时碰到的问题)

    最近在做一个php小项目的时候,发现curl调用微信的授权api.weixin.qq.com,经常是需要等待很久,但是有时候却很快. 刚开始以为是网络慢问题,没去注意.后面通过打上时间日志观察发现,慢 ...

  3. LB(Load balance)负载均衡集群--{LVS-[NAT+DR]单实例实验+LVS+keeplived实验} 菜鸟入门级

    LB(Load balance)负载均衡集群 LVS-[NAT+DR]单实例实验 LVS+keeplived实验 LVS是Linux Virtual Server的简写,意即Linux虚拟服务器,是一 ...

  4. [PHP] - 性能加速 - 开启opcache

    说明 PHP 5.5+版本以上的,可以使用PHP自带的opcache开启性能加速(默认是关闭的).对于PHP 5.5以下版本的,需要使用APC加速,这里不说明,可以自行上网搜索PHP APC加速的方法 ...

  5. j技术方案

    采用.net4.0作为基础技术平台,原来是采用.net4.5的,但是后来发现.net4.5不支持Windows Server2003,所以又降为.net4.0. 1.asp.net mvc 4.0 用 ...

  6. 关于Exception

    Exception in thread "ContainerBackgroundProcessor[StandardEngine[Catalina]]"   1.java.lang ...

  7. 优化定时器NSTimer-runloop使用

    参考文档:http://www.cnblogs.com/junhuawang/p/4647559.html   - (void)viewDidLoad { [super viewDidLoad];   ...

  8. SharePrecences--(json+sharePrecences)存list 或对象

    利用Gson和SharePreference存储结构化数据 具体的步骤 这个假设有三个User对象生成一个ArrayList<User>: User user1 = new User(&q ...

  9. HBase读延迟的12种优化套路

    任何系统都会有各种各样的问题,有些是系统本身设计问题,有些却是使用姿势问题.HBase也一样,在真实生产线上大家或多或少都会遇到很多问题,有些是HBase还需要完善的,有些是我们确实对它了解太少.总结 ...

  10. 从jquery源码中看类型判断和数组的一些操作

    在深入看jquery源码中,大家会发现源码写的相当巧妙.那我今天也通过几个源码中用到的技巧来抛砖引玉,希望大家能共同研究源码之精华,不要囫囵吞枣. 1.将类数组转化成数组 我想大家首先想到的方法是fo ...