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.观察类构造函数的构造顺序,拷贝也只有一份。
 
◇与虚函数关系 
虚拟继承与虚函数有一定相似的地方,但他们之间是绝对没有任何联系的。
再想一次:虚拟继承,虚基类,虚函数。

C++虚继承的概念(转)的更多相关文章

  1. C++虚继承的概念[转]

    C++中虚拟继承的概念 为了解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类.这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数 ...

  2. C++中的继承与虚函数各种概念

    虚继承与一般继承 虚继承和一般的继承不同,一般的继承,在目前大多数的C++编译器实现的对象模型中,派生类对象会直接包含基类对象的字段.而虚继承的情况,派生类对象不会直接包含基类对象的字段,而是通过一个 ...

  3. C++对象模型:单继承,多继承,虚继承

    什么是对象模型 有两个概念可以解释C++对象模型: 语言中直接支持面向对象程序设计的部分.对于各种支持的底层实现机制. 类中成员分类 数据成员分为静态和非静态,成员函数有静态非静态以及虚函数 clas ...

  4. C++构造函数 & 拷贝构造函数 & 派生类的构造函数 & 虚继承的构造函数

    构造函数 ,是一种特殊的方法 .主要用来在创建对象时初始化对象, 即为对象成员变量赋初始值,总与new运算符一起使用在创建对象的语句中 .特别的一个类可以有多个构造函数 ,可根据其参数个数的不同或参数 ...

  5. C++学习之虚继承

    http://blog.csdn.net/wangxingbao4227/article/details/6772579 C++中虚拟继承的概念 为了解决从不同途径继承来的同名的数据成员在内存中有不同 ...

  6. C++ 深入理解 虚继承、多重继承和直接继承

    [摘要] 本文从5段代码实例出发,通过类中类的普通继承,类的虚继承,类的多重继承,多个虚函数类的普通继承.虚继承与多重继承,几个交叉概念,详细的阐释了继承.虚函数与虚继承的基本概念,深入剖析了继承于虚 ...

  7. c++虚继承与虚函数

    学习继承与多态时看到这两个概念,记录整理. 虚继承与虚函数都是用virtual关键字实现,虚继承为了防止多重继承,而虚函数为了实现多态. 是几个例子. 虚继承: class A{}; class B: ...

  8. 虚函数&纯虚函数&抽象类&虚继承

    C++ 虚函数&纯虚函数&抽象类&接口&虚基类   1. 多态 在面向对象语言中,接口的多种不同实现方式即为多态.多态是指,用父类的指针指向子类的实例(对象),然后通过 ...

  9. C++_了解虚函数的概念

    第一.先了解基本概念介绍: 虚函数.多态.继承都是紧密相关的概念.而继承是所有概念的基础: 继承的概念:是面向对象编程的三大特性之一(另外两个是:多态和封装):继承可以使得子类具有父类的属性和方法或者 ...

随机推荐

  1. PHP smarty模版引擎基本安装

    环境:  PHP5.2 以上版本 先去官网下载smarty模版引擎的库文件到你的电脑或服务器上 smarty官方网站库文件下载地址: https://www.smarty.net/download 下 ...

  2. TensorFlow——小练习:counter

    下面的例子演示了如何使用变量实现一个 简单的计数器(counter) # _*_coding:utf-8_*_ import tensorflow as tf import numpy as np # ...

  3. Balanced Lineup(ST)

    描述 For the daily milking, Farmer John's N cows (1 ≤ N ≤ 50,000) always line up in the same order. On ...

  4. onclick跳转到其他页面的几种方式

    如果是本页显示可以直接用location,方法如下: ①onclick="javascript:window.location.href='URL'" ②onclick=" ...

  5. mybatis分页方式对比

    mybatis有两种分页方法(转自:http://blog.csdn.net/leozhou13/article/details/50394242) 1.内存分页,也就是假分页.本质是查出所有的数据然 ...

  6. HDU——1009FatMouse' Trade(贪心+结构体+排序)

    FatMouse' Trade Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

  7. for语句执行顺序

    for语句的结构如下所示: for(初始化;条件;调整) { 示例语句; } 第一次顺序:初始化 ->条件->示例语句 第二次顺序:调整->条件->示例语句 第三次顺序:调整- ...

  8. ECharts学习总结(二)-----图表组件漏斗图(funnel)

    今天在学习ECharts时,想要在ECharts图表的原生态Demo中抠出漏斗图,却不知如何下手,经过一番研究,特总结如下: 首先我们需要这样做 1.拷贝出两个js文件:esl.js 和echarts ...

  9. DispatcherServlet与ContextLoaderListener的对比

    1. 从DispatcherServlet和ContextLoaderListener的初始化过程可以看出,二者分别会生成一个WebApplicationContext,且以不同的attrName注册 ...

  10. 【BZOJ4945&&UOJ317】游戏(2-sat,拓扑序)

    题意: 思路: 输出方案时有一个优秀的性质可以利用: tarjan缩点之后点所属的分量编号是原图的反的拓扑序 所以只需要在两种方案内找到所属分量编号较小的那个就行了,用来满足(i,i')那个限制 #i ...