【转载】 C++多继承中重写不同基类中相同原型的虚函数
本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数。
在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数。下面就是一个例子:
- class CBaseA
- {
- public:
- virtual void TestA();
- };
- class CBaseB
- {
- public:
- virtual void TestB();
- };
- class CDerived : public CBaseA, public CBaseB
- {
- public:
- virtual void TestA(); // 重写基类CBaseA中的虚函数TestA()
- virtual void TestB(); // 重写基类CBaseB中的虚函数TestB()
- };
- void Test()
- {
- CDerived D;
- CBaseA *pA = &D;
- CBaseB *pB = &D;
- pA->TestA(); // 调用类CDerived的TestA()函数
- pB->TestB(); // 调用类CDerived的TestB()函数
- }
可是,如果两个基类中有一个相同原型的虚函数,例如下面这样:
- class CBaseA
- {
- public:
- virtual void Test();
- };
- class CBaseB
- {
- public:
- virtual void Test();
- };
怎样在派生类中重写这两个相同原型的虚函数呢?
也许这种情况并不常见,可是这种情况却确实存在。比如说开发的时候使用的两个类库是不同的厂商提供的,或者说这两个类库是由公司的不同开发小组开发的。对前者来说,修改基类的接口是不可能的;对后者来说,修改接口的代价很大。
如果在派生类中直接重写这个虚函数,那么2个基类的Test()虚函数都将被覆盖。这样的话就只能有一个Test()的实现,而不是像前面的例子那样有不同的实现。
- class CDerived : public CBaseA, public CBaseB
- {
- public:
- virtual void Test();
- };
- void Test()
- {
- CDerived D;
- CBaseA *pA = &D;
- CBaseB *pB = &D;
- // 下面2行代码都将调用类CDerived的Test()函数
- pA->Test();
- pB->Test();
- }
为了实现第一个例子中的那样,在派生类CDerived中重写不同基类中相同原型的虚函数Test(),可以使用下面的方法。
首先,不需要对2个基类进行任何修改(在实际的开发当中,修改基类的可能性非常小)。
- class CBaseA
- {
- public:
- virtual void Test();
- };
- class CBaseB
- {
- public:
- virtual void Test();
- };
现在,为这个继承体系添加2个中间类,分别从2个基类派生。
- class CMiddleBaseA : public CBaseA
- {
- private:
- // 真正的实现函数
- // 设置为纯虚函数,在派生类里必须实现
- virtual void CBaseA_Test() = 0;
- // 改写继承下来的虚函数
- // 仅仅直接调用真正的实现函数
- virtual void Test()
- {
- CBaseA_Test();
- }
- };
- // 与类CMiddleBaseA采用相同的方法
- class CMiddleBaseB : public CBaseB
- {
- private:
- virtual void CBaseB_Test() = 0;
- virtual void Test()
- {
- CBaseB_Test();
- }
- };
然后,类CDerived以上面2个中间类作为基类来派生。分别重写上面2个基类中原型不同的纯虚函数,添加不同的实现代码。
- class CDerived : public CMiddleBaseA, public CMiddleBaseB
- {
- private:
- // 重写从中间类继承下来的虚函数
- virtual void CBaseA_Test(); // 这里实际上是重写CBaseA的Test()
- virtual void CBaseB_Test(); // 这里实际上是重写CBaseB的Test()
- };
- void Test()
- {
- CDerived D;
- CBaseA *pA = &D;
- CBaseB *pB = &D;
- // 调用类CBaseA的Test()函数
- // 由于C++多态的特性,实际上调用的是类CDervied中的CBaseA_Test()函数
- pA->Test();
- // 调用类CBaseB的Test()函数
- // 由于C++多态的特性,实际上调用的是类CDervied中的CBaseB_Test()函数
- pB->Test();
- }
现在以上面代码中的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++多继承中重写不同基类中相同原型的虚函数的更多相关文章
- C++基类的析构函数定义为虚函数的原因
1:每个析构函数只会清理自己的成员(成员函数前没有virtual).2:可能是基类的指针指向派生类的对象,当析构一个指向派生类的成员的基类指针,这时程序不知道这么办,可能会造成内存的泄露,因此此时基类 ...
- cc31a_demo--CppPrimer_静态成员与继承-在派生类中访问基类中的static成员的方法
//*基类中的static成员,在整个继承层次中只有一个实例 //*在派生类中访问基类中的static成员的方法 //1.基类名::成员名 //2.子类名::成员名 //3.对象.成员名 //4.指针 ...
- 面向对象编程(九)——面向对象三大特性之继承以及重写、Object类的介绍
面向对象三大特性 面向对象三大特征:继承 :封装/隐藏 :多态(为了适应需求的多种变化,使代码变得更加通用!) 封装:主要实现了隐藏细节,对用户提供访问接口,无需关心方法的具体实现. 继承:很好的实现 ...
- Filter组件开发中的SDK基类分析
DirectShow SDK提供了一套开发Filter的基类源代码.基于这些基类开发Filter将大大简化开发过程. 1.CBaseObject 大部分SDK类都从CBaseObject类(参见com ...
- 自定义继承于Page的基类
自定义继承于Page的基类:MyBasePage[校验用户是否登录,如果登录则获取用户信息,否则跳转到登录页面]============================================ ...
- 基类中定义的虚函数在派生类中重新定义时,其函数原型,包括返回类型、函数名、参数个数、参数类型及参数的先后顺序,都必须与基类中的原型完全相同 but------> 可以返回派生类对象的引用或指针
您查询的关键词是:c++primer习题15.25 以下是该网页在北京时间 2016年07月15日 02:57:08 的快照: 如果打开速度慢,可以尝试快速版:如果想更新或删除快照,可以投诉快照. ...
- 读书笔记 effective c++ Item 43 了解如何访问模板化基类中的名字
1. 问题的引入——派生类不会发现模板基类中的名字 假设我们需要写一个应用,使用它可以为不同的公司发送消息.消息可以以加密或者明文(未加密)的方式被发送.如果在编译阶段我们有足够的信息来确定哪个信息会 ...
- C++可继承的单例基类模板
目录 一.介绍 二.代码 三.关键处 五.参考资料 一.介绍 最近在写一个项目,其中用到好几个单例,类本身的设计不是很复杂,但是如果每个都写一遍单例又觉得有点冗余:所以查资料写了一个单例基类模板,只要 ...
- why pure virtual function has definition 为什么可以在基类中实现纯虚函数
看了会音频,无意搜到一个frameworks/base/include/utils/Flattenable.h : virtual ~Flattenable() = 0; 所以查了下“纯虚函数定义实现 ...
随机推荐
- 获取与Url链接相关的信息
以下结果的值以此示例为基础:http://www.x2y2.com:80/fisker/post/0703/window.location.html?ver=1.0&id=6#imhere j ...
- Ill-conditioned covariance create
http://www.mathworks.com/matlabcentral/answers/100210-why-do-i-receive-an-error-while-trying-to-gene ...
- centos6.5 安装mysql 的过程
Linux中使用最广泛的数据库就是MySQL,使用在线yum的方式安装的版本落后MySQL网站好几个小版本, 所以折腾了几个星期,终于在CentOS 装好了mysql5.6,装完之后,对整个linux ...
- nginx-my
#user nobody; 3 worker_processes 1; 4 5 error_log logs/error.log; 6 #error_log logs/error.log notice ...
- CSS盒子模型与box-sizing
今天在学习的时候偶然看到一张图片: 我瞬间瞪大了眼睛:width和height竟然不包括padding和border!! 过去所学知识有问题!在我的印象里,width应该是包含padding和bord ...
- mybatis 中${}和#{}区别
用#传入参数是,sql语句解析是会加上"",比如 select * from table where name = #{name} ,传入的name为小李,那么最后打印出来的就是 ...
- TP-link TL-WN725 USB无线网卡在DX2 CPU下的Xlinux 驱动移植
网上查到,TP-link TL-WN725 USB无线网卡用的芯片是8188EU 下载驱动:http://download.csdn.net/detail/zzz_mraz/5262010 tar z ...
- VC++6.0 配置CppUTest测试环境
最近看<软件项目成功之道>,书中无数次提及到“单元测试”对于项目成败的重要性,看到同事将CppUTest用于Linux动态库测试,于是在VC++6.0环境下搭建一个基于CppUTest的单 ...
- How to pronounce symbols on keyboard
Refefrence: http://answers.yahoo.com/question/index?qid=20100607151104AAtQxhc ~ “tilde” or “tweedle” ...
- XMLHttpRequest简单总结
一.概念 XMLHttpRequest 对象用于在后台与服务器交换数据. XMLHttpRequest 对象是能够: 在不重新加载页面的情况下更新网页 在页面已加载后从服务器请求数据 在页面已加载后从 ...