首先先看一段代码:
#include<iostream>
class Base
{
public:
Base(int a):ma(a){}
void Show()
{
std::cout << "ma:" << ma << std::endl;
}
protected:
int ma;
};
class Derive :public Base
{
public:
Derive(int b):mb(b),Base(b){}
void Show()
{
std::cout << "mb:" << mb << std::endl;
}
protected:
int mb;
};
int main()
{
Base* pb = new Derive(10);
std::cout << sizeof(Base) << std::endl;
std::cout << sizeof(Derive) << std::endl;
std::cout << typeid(pb).name() << std::endl;
std::cout << typeid(*pb).name() << std::endl;
pb->Show();
return 0;
}
运行结果为:
在基类Show()函数前加上virtual关键字,结果如下:
两次操作结果比较:
virtual 虚函数
C++三大特点:封装,继承,多态
虚函数就是多态的应用。
多态:同一接口,不同形态
多态分为静多态、动多态和宏多态
静多态:编译阶段确定函数调用
动多态:运行阶段确定函数调用
宏多态:预编译阶段确定函数调用
虚函数为动多态提供支持
动多态在运行阶段如何确定函数调用?
虚函数的处理过程:
首先,在编译阶段,生成完符号之后,把虚函数的符号不仅给符号表中放一份也给只读数据段中放一份,只读数据段中的数据可以用一个结构体来表示,叫做vftable,虚函数表在编译阶段生成,当只读数据段合并完成好之后,EF文件中也会存在一个只读数据段,存放了虚函数表的信息,表上有虚函数的入口地址。