C++面向对象中,虚函数与虚继承是两个完全不同的概念。

一、虚函数

C++程序中只要类中含有虚拟函数,编译程序都会为此类生成一个对应的虚拟函数跳转表(vtbl),该虚拟函数跳转表是一个又若干个虚拟函数体入口地址组成的一个线性表。派生类的虚拟函数跳转表的前半部分由父类的vtbl得出,但是里面的内容不一定相同,后半部分则对应着自己新定义的虚拟函数。

class Employee
{
protected:
char *Name;
int Age;
public:
void changeAge(int newAge);
virtual void retire(); //虚函数
Employee( char *n, int a );
~Employee();
}; class Manager: public Employee
{
int Level;
public:
void changeLevel( int l);
void retire(); //情况一、子类覆写了虚函数
Manager( char *n, int a, int l );
~Manager();
};

情况一、子类覆写了父类的虚函数,则此时子类和父类的虚函数表分别为:

情况二、子类没有覆写父类的虚函数,则此时子类和父类的虚函数表分别为:

class Manager: public Employee
{
int Level;
public:
void changeLevel( int l);
//void retire(); //情况二、子类没有覆写虚函数
Manager( char *n, int a, int l );
~Manager();
};

情况三、子类中自己的虚函数,则此时子类和父类的虚函数表分别为:

class Manager: public Employee
{
int Level;
public:virtual void changeLevel(int l); //情况三、子类有自己的虚拟函数
void retire();
Manager( char *n, int a, int l );
~Manager();
};

二、虚继承

虚继承是为了解决多重继承中的问题而出现的。要理解虚继承的实现机制,首先看一般继承:

class A
{
char k[];
public:
virtual void aa()
{ };
};
class B:public A
{
char j[];
public:
virtual void bb()
{ };
};
class C:public B
{
char i[];
public:
virtual void cc()
{ };
};
int main()
{
cout<<sizeof(A)<<endl;
cout<<sizeof(B)<<endl;
cout<<sizeof(C)<<endl;
return ;
}

得到的答案是:8 12 16

分析如下:

1、  对于类A,有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量char k[3],则根据数据对齐原则,可以得到sizeof(A)的大小为8;

2、   对于类B,也有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量j[3],但是类B是继承自类A的,是一般继承,不是虚继承,所以可以得到sizeof(B)的大小为:  4(指向虚函数的虚指针)+4(自己的数据成员)+4(父类A的数据成员)=12;

3、  对于类C,也有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量i[3],同样的,类C是继承自类B的,是一般继承,不是虚继承,所以可以得到sizeof(C)的大小为4(指向虚函数的虚指针)+4(自己的数据成员)+8(父类的数据成员)=16;

再来看一下虚继承的情况:

class A
{
char k[];
public:
virtual void aa()
{ };
};
class B:public virtual A
{
char j[];
public:
virtual void bb()
{ };
};
class C:public virtual B
{
char i[];
public:
virtual void cc()
{ };
};
int main()
{
cout<<sizeof(A)<<endl;
cout<<sizeof(B)<<endl;
cout<<sizeof(C)<<endl;
return ;
}

得到的答案是:8 20 32

分析如下:

1、对于类A,有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量k[3],则根据数据对齐原则,可以得到sizeof(A)的大小为8;

2、对于类B,也有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量j[3],但是类B是继承自类A的,而且是虚继承,虚继承的实现是通过一个虚基类指针列表,比如vptr_B_A指向虚基类,同时还包含了父类的所有内容,所以可以得到sizeof(B)的大小为:4(指向自己虚函数的虚指针)+4(自己的数据成员)+4(指向虚基类的指针vptr_B_A)+8(父类A的内容大小)=20;

3、对于类C,也有一个虚函数,需要一个虚函数表vtbl来记录函数的入口地址,每个地址一个虚指针,指针的大小为4,类中还有一个成员变量i[3],同样的,类C是继承自类B的,而且是虚继承,虚继承的实现是通过一个虚基类指针列表,比如vptr_C_B指向虚基类,同时还包含了父类的所有内容,所以可以得到sizeof(C)的大小为:4(指向自己虚函数的虚指针)+4(自己的数据成员)+4(指向虚基类的指针(vptr_C_B)+20(父类B的内容大小)=32;

class A
{
char k[];
public:
virtual void aa()
{ };
};
class B:public virtual A
{
char j[];
public:
virtual void bb()
{ };
};
class C:public virtual A
{
char i[];
public:
virtual void cc()
{ };
};
class D: public B,public C
{
char n[];
public:
virtual void dd()
{ };
};
int main()
{
cout<<sizeof(A)<<endl;
cout<<sizeof(B)<<endl;
cout<<sizeof(C)<<endl;
cout<<sizeof(D)<<endl;
return ;
}

得到的结果为:8 20 20 36

分析如下:

1、类sizeof(A,B,C),从前面的分析可以得到结果为8 20 20;

2、sizeof(D)=4(指向自己虚函数的虚指针)+4(自己的数据成员)+

12(父类B的内容,包括数据成员、指向虚函数的虚指针、以及虚基类指针列表vptr_B_A)+

12(父类C的内容,包括数据成员、指向虚函数的虚指针、以及虚基类指针列表vptr_C_A)+

4(类A的数据成员)=36。

     可见,类D中只包含了类A的一份副本,虚继承很好地解决了多重继承中的二义性问题。

C++之易混淆知识点四---虚函数与虚继承的更多相关文章

  1. Java-web易混淆知识点整理

    Java-web易混淆知识点 post和get区别 post: 数据不会显示在地址栏 安全 大小无限制 可以提交二进制文件 get: 数据显示在地址栏 不安全 get方式提交有大小限制(约4kb) 相 ...

  2. C++虚函数及虚函数表解析

    一.背景知识(一些基本概念) 虚函数(Virtual Function):在基类中声明为 virtual 并在一个或多个派生类中被重新定义的成员函数.纯虚函数(Pure Virtual Functio ...

  3. C++基础知识 基类指针、虚函数、多态性、纯虚函数、虚析构

    一.基类指针.派生类指针 父类指针可以new一个子类对象 二.虚函数 有没有一个解决方法,使我们只定义一个对象指针,就可以调用父类,以及各个子类的同名函数? 有解决方案,这个对象指针必须是一个父类类型 ...

  4. virtual之虚函数,虚继承

    当类中包含虚函数时,则该类每个对象中在内存分配中除去数据外还包含了一个虚函数表指针(vfptr),指向虚函数表(vftable),虚函数表中存放了该类包含的虚函数的地址. 当子类通过虚继承的方式从父类 ...

  5. C++纯虚函数、虚函数、实函数、抽象类,重载、重写、重定义

    首先,面向对象程序设计(object-oriented programming)的核心思想是数据抽象.继承.动态绑定.通过数据抽象,可以使类的接口与实现分离,使用继承,可以更容易地定义与其他类相似但不 ...

  6. C++之虚函数与虚继承详解

    准备工作 1.VS2012使用命令行选项查看对象的内存布局 微软的Visual Studio提供给用户显示C++对象在内存中的布局的选项:/d1reportSingleClassLayout.使用方法 ...

  7. C++ 由虚基类 虚继承 虚函数 到 虚函数表

    //虚基类:一个类可以在一个类族中既被用作虚基类,也被用作非虚基类. class Base1{ public: Base1(){cout<<"Construct Base1!&q ...

  8. 为何JAVA虚函数(虚方法)会造成父类可以"访问"子类的假象?

      首先,来看一个简单的JAVA类,Base. 1 public class Base { 2 String str = "Base string"; 3 protected vo ...

  9. C++虚函数和虚函数表

    前导 在上面的博文中描述了基类中存在虚函数时,基类和派生类中虚函数表的结构. 在派生类也定义了虚函数时,函数表又是怎样的结构呢? 先看下面的示例代码: #include <iostream> ...

随机推荐

  1. hdu 2768 Cat vs. Dog 最大独立集 巧妙的建图

    题目分析: 一个人要不是爱狗讨厌猫的人,要不就是爱猫讨厌狗的人.一个人喜欢的动物如果离开,那么他也将离开.问最多留下多少人. 思路: 爱猫和爱狗的人是两个独立的集合.若两个人喜欢和讨厌的动物是一样的, ...

  2. 从无到有创建一个grunt项目

    在安装好grunt的前提下创建一个grunt的项目: 1.首先创建一个项目文件 就叫grunt-project 2.进入这个文件 创建一个index.html 在创建一个js文件,进去创建一个inde ...

  3. CentOS6.6安装mysql-5.7.25二进制安装包简易教程

    #####1,安装前首先确认系统版本 [root@bogon:~]# cat /etc/redhat-release CentOS release 6.6 (Final) [root@bogon:~] ...

  4. MySQL py模块的链接Navicat可视化工具

     数据库可视化工具Navicat 1 基本操作: 1 库 表 字段 记录(增删改查) 2 添加主建,添加自增. 3 添加外键,外键的链接 4 模型建表,模型添加外键.(逆向数据库到模型,运行SQL文件 ...

  5. React传递参数的多种方式

    最常见的就是父子组件之间传递参数 父组件往子组件传值,直接用this.props就可以实现 在父组件中,给需要传递数据的子组件添加一个自定义属性,在子组件中通过this.props就可以获取到父组件传 ...

  6. 计蒜客 阿里天池的新任务—简单( KMP水 )

    链接:传送门 思路:KMP模板题,直接生成 S 串,然后匹配一下 P 串在 S 串出现的次数,注意处理嵌套的情况即可,嵌套的情况即 S = "aaaaaa" ,P = " ...

  7. C语言求大数的阶乘

    我们都知道如何计算一个数的阶乘,可是,如果这个数很大呢,该如何计算? 当一个数很大时,利用平常的方法是求不出来它的阶乘的,因为数据超出了范围.因此我们要用数组来求一个大数的阶乘,用数组的每位表示结果的 ...

  8. HDU 5126 stars (四维偏序+树状数组)

    题目大意:略 题目传送门 四维偏序板子题 把插入操作和询问操作抽象成$(x,y,z,t)$这样的四元组 询问操作拆分成八个询问容斥 此外$x,y,z$可能很大,需要离散 直接处理四维偏序很困难,考虑降 ...

  9. 「Poetize4」创世纪

    在tyvj上怀疑爆栈了.....或许一定是我写挂了.以后调吧... UPD:bzoj上过了... 题解:https://blog.csdn.net/popoqqq/article/details/39 ...

  10. 【hihocoder 1312】搜索三·启发式搜索(启发式搜索写法)

    [题目链接]:http://hihocoder.com/problemset/problem/1312?sid=1092363 [题意] [题解] 定义一个A*函数 f = step+val 这里的v ...