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. 【整理】python中re的match、search、findall、finditer区别

    match 从首字母开始开始匹配,string如果包含pattern子串,则匹配成功,返回Match对象,失败则返回None,若要完全匹配,pattern要以$结尾. search 若string中包 ...

  2. javascript基础 方法 函数 闭包 集合

    定义类 ,实例化对象类 ,调用 为类对象增加数据成员 --

  3. redhat linux 从/home目录扩展空间至/根目录

    查看分区大小 [root@easdb01 ~]# df -hFilesystem Size Used Avail Use% Mounted on/dev/mapper/vg_easdb01-lv_ro ...

  4. [AtCoderContest075F]Mirrored

    [AtCoderContest075F]Mirrored 试题描述 For a positive integer \(n\), we denote the integer obtained by re ...

  5. 修改Tomcat HTTP端口号(8080→8088)Eclipse

    1.如果不用Eclipse,修改Tomcat的端口号,可以参考这篇: https://jingyan.baidu.com/article/adc815139b12def722bf7377.html t ...

  6. [HDU4362] Palindrome subsequence (区间DP)

    题目链接 题目大意 给你几个字符串 (1<len(s)<1000) ,要你求每个字符串的回文序列个数.对于10008取模. Solution 区间DP. 比较典型的例题. 状态定义: 令 ...

  7. bzoj 3143 [Hnoi2013]游走 期望dp+高斯消元

    [Hnoi2013]游走 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3394  Solved: 1493[Submit][Status][Disc ...

  8. bzoj3997[TJOI2015]组合数学(求最长反链的dp)

    组合数学 给出一个网格图,其中某些格子有财宝,每次从左上角出发,只能向下或右走.问至少走多少次才能将财宝捡完.此对此问题变形,假设每个格子中有好多财宝,而每一次经过一个格子至多只能捡走一块财宝,至少走 ...

  9. 【CF1029A】Many Equal Substrings(模拟)

    题意:给定一个长度为n的串s,要求构造一个长度最小的使s出现了k次的串,可以重叠 n<=50,k<=50 思路:计算一下前后缀相同长度 #include<cstdio> #in ...

  10. 卸载重安firefox

    把firefox完全卸载掉重装: 查看安装的firefox版本: dpkg --get-selections |grep firefox 根据命令结果卸载重装 比如: $ dpkg --get-sel ...