c++虚函数表

C++中的虚函数通过虚函数表(V-Table)实现,解决继承和覆盖问题,保证正确调用实际函数。虚函数表包含类的虚函数地址,对象实例中存储指向该表的指针。编译器通常将虚表指针放在对象实例开始位置,便于高效访问。文章探讨了单继承(有无覆盖)、多继承(有无覆盖)的虚表布局和行为。
摘要由CSDN通过智能技术生成

C++中的虚函数(Virtual Function)的实现通过一张虚函数表来实现的,简称为V-Table。在这个表中,主要是一个类的虚函数的地址表,这张表解决了继承、覆盖的问题,保证其真实反应实际的函数。这样,在有虚函数的类的实例中分配了指向这个表的指针的内存,所以,当用父类的指针来操作一个子类的时候,这张虚函数表就显得尤为重要了,它就像一个地图一样,指明了实际所应该调用的函数。

编译器应该是保证虚函数表的指针存在于对象实例中最前面的位置(这是为了保证取到虚函数表的有最高的性能——如果有多层继承或是多重继承的情况下)。这意味着可以通过对象实例的地址得到这张虚函数表,然后就可以遍历其中函数指针,并调用相应的函数。

下面讨论虚函数表的各种情况,我用的是VS 2013,因此都是在VS 2013编译器下讨论各种情况的:

一、单继承(无虚函数覆盖)

1、例子

#include<iostream>
using namespace std;

class Base {
public:
	virtual void f() { cout << "Base::f()" << endl; }
	virtual void g() { cout << "Base::g()" << endl; }
	virtual void h() { cout << "Base::h()" << endl; }
private:
	virtual void i() { cout << "Base::i()" << endl; }
};
class Derive:public Base{
public:
	virtual void f1(){ cout << "Derive::f1()" << endl; }
	virtual void g1(){ cout << "Derive::g1()" << endl; }
	virtual void h1(){ cout << "Derive::h1()" << endl; }
};

int main(){
    typedef void(*pFun)(); //定义一个函数指针,参数列表空,返回值void
	Base base;
	int** pVtab = (int**)&base;
	for (int i = 0; pVtab[0][i] != NULL; ++i){
		cout << "虚函数表存放第"<<i+1<<"个函数地址的地址:" << pVtab[0]+i << endl; //pVtab[0]+i等同于(int*)*(int*)(&base) + i,参考文章那边有错误 
		cout << "虚函数表第" << i + 1 << "个函数地址:" << (int*)pVtab[0][i] << endl;//(int*)pVtab[0][i]等同于(int*)*((int*)*(int*)(&base) + 1)
		((pFun)pVtab[0][i])();
	}
	cout << "------------------------------------------------------------------" << endl;
	Derive derive;
	pVtab = (int**)&derive;
	for (int i = 0; pVtab[0][i] != NULL; ++i){
		cout << "虚函数表存放第" << i + 1 << "个函数地址的地址:" << pVtab[0] + i << endl; //pVtab[0]+i等同于(int*)*(int*)(&derive) + i
		cout << "虚函数表第" << i + 1 << "个函数地址:" << (int*)pVtab[0][i] << endl; //(int*)pVtab[0][i]等同于(int*)*((int*)*(int*)(&derive) + 1)
		((pFun)pVtab[0][i])();
	}

	system("pause");
	return 0; 
}

2、运行结果:


3、类继承关系


4、基类虚表指针及虚表布局:


5、继承类虚表指针及虚表布局:

<

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值