http://blog.csdn.net/wangxingbao4227/article/details/6772579

C++中虚拟继承的概念

为了解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类。这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。这样不仅就解决了二义性问题,也节省了内存,避免了数据不一致的问题。
class 派生类名:virtual 继承方式  基类名
virtual是关键字,声明该基类为派生类的虚基类。
在多继承情况下,虚基类关键字的作用范围和继承方式关键字相同,只对紧跟其后的基类起作用。
声明了虚基类之后,虚基类在进一步派生过程中始终和派生类一起,维护同一个基类子对象的拷贝。

C++虚拟继承

◇概念:

C++使用虚拟继承(Virtual Inheritance),解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类。这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。

◇解决问题:

解决了二义性问题,也节省了内存,避免了数据不一致的问题。
 
◇同义词: 
虚基类(把一个动词当成一个名词而已)
当在多条继承路径上有一个公共的基类,在这些路径中的某几条汇合处,这个公共的基类就会产生多个实例(或多个副本),若只想保存这个基类的一个实例,可以将这个公共基类说明为虚基类。

◇语法:

 class 派生类: virtual 基类1,virtual 基类2,...,virtual 基类n

 {

 ...//派生类成员声明

 };

◇执行顺序

首先执行虚基类的构造函数,多个虚基类的构造函数按照被继承的顺序构造;

执行基类的构造函数,多个基类的构造函数按照被继承的顺序构造;

执行成员对象的构造函数,多个成员对象的构造函数按照申明的顺序构造;

执行派生类自己的构造函数;

析构以与构造相反的顺序执行;

mark

从虚基类直接或间接派生的派生类中的构造函数的成员初始化列表中都要列出对虚基类构造函数的调用。但只有用于建立对象的最派生类的构造函数调用虚基类的构造函数,而该派生类的所有基类中列出的对虚基类的构造函数的调用在执行中被忽略,从而保证对虚基类子对象只初始化一次。

在一个成员初始化列表中同时出现对虚基类和非虚基类构造函数的调用时,虚基类的构造函数先于非虚基类的构造函数执行。

◇因果:

多重继承->二义性->虚拟继承解决

◇二义性:

 : //-----------------------------------------------------
: //名称:blog_virtual_inherit.cpp
: //说明:C++虚拟继承学习演示
: //环境:VS2005
: //blog:pppboy.blog.163.com
: //----------------------------------------------------
: #include "stdafx.h"
: #include <iostream>
: using namespace std;
:
: //Base
: class Base
: {
: public:
: Base(){cout << "Base called..."<< endl;}
: void print(){cout << "Base print..." <<endl;}
: private:
: };
:
: //Sub
: class Sub //定义一个类 Sub
: {
: public:
: Sub(){cout << "Sub called..." << endl;}
: void print(){cout << "Sub print..." << endl;}
: private:
: };
:
: //Child
: class Child : public Base , public Sub //定义一个类Child 分别继承自 Base ,Sub
: {
: public:
: Child(){cout << "Child called..." << endl;}
: private:
: };
:
: int main(int argc, char* argv[])
: {
: Child c;
:
: //不能这样使用,会产生二意性,VC下error C2385
: //c.print();
:
: //只能这样使用
: c.Base::print();
: c.Sub::print();
:
: system("pause");
: return ;
: }

◇多重继承:

 
 : //-----------------------------------------------------
: //名称:blog_virtual_inherit.cpp : //说明:C++虚拟继承学习演示
: //环境:VS2005
: //blog:pppboy.blog.163.com
: //----------------------------------------------------
: #include "stdafx.h"
: #include <iostream>
: using namespace std;
:
: int gFlag = ;
:
: class Base
: {
: public:
: Base(){cout << "Base called : " << gFlag++ << endl;}
: void print(){cout << "Base print" <<endl;}
: };
:
: class Mid1 : public Base
: {
: public:
: Mid1(){cout << "Mid1 called" << endl;}
: private:
: };
:
: class Mid2 : public Base
: {
: public:
: Mid2(){cout << "Mid2 called" << endl;}
: };
:
: class Child:public Mid1, public Mid2
: {
: public:
: Child(){cout << "Child called" << endl;}
: };
:
: int main(int argc, char* argv[])
: {
: Child d;
:
: //不能这样使用,会产生二意性
//d.print();
:
: //只能这样使用
: d.Mid1::print();
: d.Mid2::print();
:
: system("pause");
: return ;
: }
:

//output

 
 Base called : 0
 Mid1 called
 Base called : 1 
 Mid2 called
 Child called
 Base print
 Base print

◇虚拟继承

在派生类继承基类时,加上一个virtual关键词则为虚拟继承

 
: //-----------------------------------------------------
: //名称:blog_virtual_inherit.cpp : //说明:C++虚拟继承学习演示
: //环境:VS2005
: //blog:pppboy.blog.163.com
: //----------------------------------------------------
: #include "stdafx.h"
: #include <iostream>
: using namespace std;
:
: int gFlag = ;
:
: class Base
: {
: public:
: Base(){cout << "Base called : " << gFlag++ << endl;}
: void print(){cout << "Base print" <<endl;}
: };
:
: class Mid1 : virtual public Base
: {
: public:
: Mid1(){cout << "Mid1 called" << endl;}
: private:
: };
:
: class Mid2 : virtual public Base
: {
: public:
: Mid2(){cout << "Mid2 called" << endl;}
: };
:
: class Child:public Mid1, public Mid2
: {
: public:
: Child(){cout << "Child called" << endl;}
: };
:
: int main(int argc, char* argv[])
: {
: Child d;
:
: //这里可以这样使用
: d.print();
:
: //也可以这样使用
: d.Mid1::print();
: d.Mid2::print();
:
: system("pause");
: return ;
: }
:

//output

 
 1: Base called : 0
 2: Mid1 called
 3: Mid2 called
 4: Child called
 5: Base print
 6: Base print
 7: Base print
 8: 请按任意键继续. . .

◇通过输出的比较

1.在多继承情况下,虚基类关键字的作用范围和继承方式关键字相同,只对紧跟其后的基类起作用。
2.声明了虚基类之后,虚基类在进一步派生过程中始终和派生类一起,维护同一个基类子对象的拷贝。
3.观察类构造函数的构造顺序,拷贝也只有一份。
 
◇与虚函数关系 
虚拟继承与虚函数有一定相似的地方,但他们之间是绝对没有任何联系的。
再想一次:虚拟继承,虚基类,虚函数。

最新文章

  1. Java Keytools 证书转换成Openssl 的PEM 文件或keytools 导出私钥文件
  2. NSString 和NSData 转换
  3. ios8版本地图定位注意点
  4. JNDI 和JDBC的区别
  5. 遭遇OutOfMemoryError
  6. 简单的JS多物体的运动---运动和透明度的变化
  7. solr python客户端 - solrpy
  8. 一个自动生成插入与更新SQL语句的小类
  9. Python入门和基础
  10. hadoop cdh 4.5的安装配置
  11. Android SurfaceView实现全屏播放例子
  12. js登录页面的 回车事件
  13. 【剑指offer】面试题42:翻转单词顺序 VS 左旋转字符串
  14. TCPWrap的使用配置
  15. 利用数据库触发器让字段与自增长Id相关联
  16. MySQL DataType--数值类型
  17. 获取微信小程序源码
  18. centos6下时间同步(ntp)操作
  19. Beta阶段第1周/共2周 Scrum立会报告+燃尽图 06
  20. SQL 查询函数

热门文章

  1. 缓存淘汰算法之LFU
  2. Struts2报错:No result defined for action xxx and result input
  3. Django多变关联、增加数据、删除数据
  4. [uiautomator篇] [4] 运行成功的日志打印---最后写一个脚本来实现
  5. 【Mysql 优化 6】mysql优化的内容和思路
  6. Android内存使用——垃圾回收LOG,GC_CONCURRENT等的意义的说明
  7. 【P2387】魔法森林(SPFA非正解)
  8. POJ——3070Fibonacci(矩阵快速幂)
  9. 广东工业大学2016校赛决赛重现——E积木积水(方法据说很多)
  10. 学习 WebService 第四步:利用WSDL(URL)生成WebService客户端&lt;初级&gt;