关于虚函数、虚继承和虚表
来源:互联网 发布:stellaris 知乎 编辑:程序博客网 时间:2024/06/06 18:57
一、虚函数
首先,虚函数的定义为在函数前添加关键字virtual。然后,之所以定义虚函数,是为了实现语言的多态性的特点。
虚函数里面有纯虚函数的玩意。通过直接在虚函数后面添加= 0来实现,举例如下:
virtual void (*Fun)() = 0;
应该注意的是,当一个类中出现了至少一个纯虚函数时,这个类就成为了传说中的抽象类。抽象类的特点是无法被实例化为一个对象。原因就涉及到了纯虚函数的特点。纯虚函数表示这个虚函数还没有实现,它要在这个类的派生类中具体的实现,然后它才可以被实例化。这个的好处就是提供接口,留着实现方法去等待实现,就像BOSS把任务接口给出,而具体的实现就让员工去实现,而当员工没有实现的时候,项目自然就无法实现了。
二、虚表
当一个类出现了虚函数时,不管是纯虚函数还是普通的虚函数,此时,类中都会出现一个vptr的虚表指针。因此,这里就会出现许多关于求类的大小的问题。特别注意,空类和只有普通类成员函数的类的大小为1,而出现纯虚函数和虚函数时,则增加一个指针类型的大小。这里给出求虚表地址的求法。先给出代码:
#include<iostream>using namespace std;typedef void(*Fun)(void);class A{public:virtual void fun1(){cout << "this is fun1." << endl;}virtual void fun2(){cout << "this is fun2." << endl;}private:int x;};int main(){A a;printf("%p\n",&a);printf("%p\n", (int *)*(int *)&a);cout << (int *)&a << endl;cout << (int *)*(int *)&a << endl;Fun p = (Fun)*((int *)*(int *)(&a)+1);p();p = (Fun)*((int *)*(int *)&a);p();((Fun)*((int *)*(int *)(&a) + 1))();return 0;}*/
首先,我们定义了一个包含虚函数的对象a。&a表示取得对象的地址,但是对象的地址不等于虚表的地址,虽然虚表位于对象的存储块的最开始位置。我们可以通过(int*)&a取得对象首元素的地址,即第一张虚表的位置。然后通过*引用,获取虚表的第一个元素,即指向fun1()的函数指针,此时,通过(int*)来强转成整形指针地址类型,因为一个指针的大小为4字节,而后通过+1操作获取下一个虚函数的地址指针。所以,(int *)&a为虚表的地址,(int *)*(int *)&a为第一个虚函数的地址。当需要调用这个虚函数时,先*引用这个虚函数指针,再加上(),如*(int*)*(int*)&a。
三、虚继承
首先,分析简单的虚继承问题。当类中出现了纯虚函数或虚函数时,此时派生类都会有两种情况:派生类原来没有虚表,继承了父类的虚表;派生类原来有虚表,此时父类的虚表替换子类虚表,将子类虚表置入父表中,然后子类中的虚函数覆盖父类中的同名虚函数。
- 关于继承和虚函数的问题
- 关于虚函数、虚继承和虚表
- 继承和虚函数
- 关于虚函数与继承
- 关于求虚函数的和虚继承中的大小
- 一个关于继承和虚函数的问题
- 虚函数继承和虚继承
- 虚继承和虚函数继承
- 虚继承和虚函数继承
- 虚继承和虚函数继承
- 继承----有关虚函数和虚拟继承
- 虚函数表指针和虚继承
- 解析虚函数表和虚继承
- 虚继承和虚函数
- 虚函数和虚继承
- 继承和纯虚函数
- 多重继承和虚函数
- 菱形继承和虚函数
- C语言strstr()函数:返回字符串中首次出现子串的地址
- BZOJ4031——HEOI小z的房间
- 并发 并行 同步 异步 多线程 阻塞 非阻塞的区别
- struts2中s:radio标签的使用
- 最长上升子序列n*n和nlogn 算法
- 关于虚函数、虚继承和虚表
- 【2015.07.13周一】【Thinkphp】【URL重写】隐藏index.php的步骤
- 寻址方式
- mel笔记
- Android Studio系列教程二--基本设置与运行
- Android Activity向右滑动返回
- iOS上Sqlite多线程问题
- shell学习四十六天----文件系统的空间信息df和du命令
- js 获取两个标准时间差