C++ 多态、虚函数机制以及虚函数表
1.非virtual函数,调用规则取决于对象的显式类型。例如
A* a = new B();
a->display();
调用的就是A类中定义的display()。和对象本体是B无关系。
2.virtual函数,具体调用哪个版本,取决于虚函数表。例如
A* a = new B();
a->v_display();
这个时候,对象a就需要查找自身的虚函数表,表中的v_display()是一个函数指针,可能指向不同类中的对应的v_display函数并调用对应版本的v_display。一般而言,如果B重写了v_display(),那么会调用B的v_display(),如果没有会调用A的display();
3.纯虚函数。结合2分析,纯虚函数在父类是没有定义的,如果子类继承后仍然不重写并定义对应的纯虚函数,那么子类也是抽象类,无法初始化对象,直到有一个类定义了这个虚函数。
这个类才可以初始化对象。并在这个对象上维护这个函数的虚函数表相关数据。
关于虚函数表:
一般存储在一个对象的固定的地址,我们甚至可以通过虚函数表来直接调用某些虚函数。
例如:
A* a = new B();
a就是对象a的地址
将a的地址强制转化成int型指针 (int*)a ,也就是理想化将对象看成int型数组。
而上述这些地址中每个地址解引用会获得什么呢?别的不知道,第一个存储的就是虚函数表,里边又是一组地址!我们将第一个地址解引用:
*(int*)(a)
然后得到了一个存储虚函数表指针的首地址(如果直接打印会打印的int十进制),然后再将其强转成int*,把这个十进制的数值看成一个int*型指针,这样虚函数表的第一个地址我们也得到了,那就是
(int*)*(int*)(a)
这个地址可以像int[] 一样计算 ,如果+1 就可以获得虚函数表的下一个函数。但是毕竟得到的都是每个函数的地址,所以我们还需要将它解引用:
*(int*)*(int*)(a) 这样就获得真实的函数指针了!
到这里就简单了,假如我们调用的函数是void(void)类型的,我们创建一个函数指针类型:
typedef void(*Fun)(void)
这样如果上边从虚函数表中获得的函数指针也是void(void)类型的,我们就可以将其强转为对应的函数指针类型
即
Fun vfun = (Fun)*(int*)*(int*)(a);
vfun();
哦卖萧的。。。。。。
我这里有一组测试代码:
#include <iostream> #include <string> using namespace std; typedef void(*Fun)(void); class A { private : string s; public : virtual void display(); virtual void v_display(); ; }; class B : public A { public: void display(); //void v_display(); void v_0_display () const; }; void A::display() { cout << "A::display" << endl; } void A::v_display() { cout << "A::v_display" << endl; } void B::display() { cout << "B::display" << endl; } //void B::v_display() //{ // cout << "B::v_display" << endl; //} void B::v_0_display() const { cout << "B::v_0_display" << endl; } int main(int argc, char* argv[]) { A* a = new B(); B* b = new B(); Fun vfun = NULL; vfun = (Fun)*((); cout << ) << endl; cout << "------------virtual table-------------" << endl; vfun(); vfun = (Fun)*((); vfun(); vfun = (Fun)*((); vfun(); cout << "--------------------------------------" << endl; a->display(); a->v_display(); a->v_0_display(); ((B*)(a))->display(); ((B*)(a))->v_display(); ((B*)(a))->v_0_display(); b->display(); b->v_display(); b->v_0_display(); ((A*)(b))->display(); ((A*)(b))->v_display(); ((A*)(b))->v_0_display(); system("pause"); }
打印结果为:
the address the first virtual function is002FDC94 ------------virtual table------------- B::display A::v_display B::v_0_display -------------------------------------- B::display A::v_display B::v_0_display B::display A::v_display B::v_0_display B::display A::v_display B::v_0_display B::display A::v_display B::v_0_display 请按任意键继续. . .
C++ 多态、虚函数机制以及虚函数表的更多相关文章
- C++中对C的扩展学习新增内容———面向对象(继承)函数扩展性及虚函数机制
1.c语言中的多态,动态绑定和静态绑定 void do_speak(void(*speak)()) { speak(); } void pig_speak() { cout << &quo ...
- 浅谈C++虚函数机制
0.前言 在后端面试中语言特性的掌握直接决定面试成败,C++语言一直在增加很多新特性来提高使用者的便利性,但是每种特性都有复杂的背后实现,充分理解实现原理和设计原因,才能更好地掌握这种新特性. 只要出 ...
- [C/C++] 虚函数机制
转自:c++ 虚函数的实现机制:笔记 1.c++实现多态的方法 其实很多人都知道,虚函数在c++中的实现机制就是用虚表和虚指针,但是具体是怎样的呢?从more effecive c++其中一篇文章里面 ...
- C++ 类的多态三(多态的原理--虚函数指针--子类虚函数指针初始化)
//多态的原理--虚函数指针--子类虚函数指针初始化 #include<iostream> using namespace std; /* 多态的实现原理(有自己猜想部分) 基础知识: 类 ...
- 匹夫细说C#:从园友留言到动手实现C#虚函数机制
前言 上一篇文章匹夫通过CIL代码简析了一下C#函数调用的话题.虽然点击进来的童鞋并不如匹夫预料的那么多,但也还是有一些挺有质量的来自园友的回复.这不,就有一个园友提出了这样一个代码,这段代码如果被编 ...
- C++多态、虚函数、纯虚函数、抽象类、虚基类
一.C++多态 C++的多态包括静态多态和动态多态.静态多态包括函数重载和泛型编程,动态多态包括虚函数.静态多态是指在编译期间就可以确定,动态多态是指在程序运行时才能确定. 二.虚函数 1.虚函数为类 ...
- c++ 虚函数和纯虚函数
在你设计一个基类的时候,如果发现一个函数需要在派生类里有不同的表现,那么它就应该是虚的.从设计的角度讲,出现在基类中的虚函数是接口,出现在派生类中的虚函数是接口的具体实现.通过这样的方法,就可以将对象 ...
- C++ - 虚基类、虚函数与纯虚函数
虚基类 在说明其作用前先看一段代码 class A{public: int iValue;}; class B:public A{public: void bPrintf(){ ...
- C++ 虚函数 、纯虚函数、接口的实用方法和意义
也许之前我很少写代码,更很少写面向对象的代码,即使有写多半也很容易写回到面向过程的老路上去.在写面向过程的代码的时候,根本不管什么函数重载和覆盖,想到要什么功能就变得法子的换个函数名字,心里想想:反正 ...
随机推荐
- 剑指offer——树的子结构 (JAVA代码)
版权声明:本文为博主原创文章,未经博主允许不得转载. 题目描述: 输入两棵二叉树A,B,判断B是不是A的子结构.(ps:我们约定空树不是任意一个树的子结构). 解题思路: 首先看牛客网给出的测试用例: ...
- 移动端接口:java写get方式访问数据(springmvc+spring。。。)
很多时候,一个问题想明白,找对点了再去问,这样被问的人也知道怎么给你讲,你也听的明白. 就比如做移动端接口,上去就问 怎么弄接口呀,其实是没找到主要的点上,所以不知道怎么弄,那个点就是手机接口是干嘛的 ...
- Codeforces Round #342 (Div. 2)-B. War of the Corporations
time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...
- STM32 MX Cube生成的工程中 使用printf向Uart发送数据
1. 在main函数前面添加: int fputc(int ch,FILE *f){ uint8_t temp[1]={ch}; HAL_UART_Transmit(&huart1,temp, ...
- c++的多线程和多进程
一.多进程和多线程对比 多进程:进程不止一个,开销比较大,通信方式比较复杂(可以用过管道.文件.消息队列进行通信),维护成本不高. 多线程:利用共享内存的方式进行指令的执行,开销比较低,但是维护起来比 ...
- tinkphp登录验证码的使用
登录和验证(控制器) <?php namespace Come\Controller; use Think\Controller; class RenController extends Con ...
- Web Performance Test : 为Request的Post参数名添加XPath支持
问题描述 本文的标题看起来有些含糊其辞,这里我需要把问题阐述得更加清楚.这是我们使用VSTS进行Web Performance Test时,Asp.net造成的特定问题(也许其他开发工具或插件也会造成 ...
- 24SQL
create or replace function f_op (p_n1 number , p_op varchar2 , p_n2 number ) return number a ...
- GOLANG 常用命令
golang常用命令: 命令 功能 build 编译包和依赖 run 编译并且直接运行 install 编译安装包和依赖 get 下载并安装包和依赖 fmt 调用gofmt格式化源码文件 d ...
- 利用PhantomJS进行网页截屏,完美解决截取高度的问题
关于PhantomJS PhantomJS 是一个基于WebKit的服务器端 JavaScript API.它全面支持web而不需浏览器支持,其快速,原生支持各种Web标准: DOM 处理, CSS ...