#include
class A
{
public:
void f1()
{
std::cout << "A::f1 " << std::endl;
}
virtual void f2()
{
std::cout << "A::f2 " << std::endl;
}
};
class B
{
public:
virtual void f1()
{
std::cout << "B::f1 " << std::endl;
}
virtual void f2()
{
std::cout << "B::f2 " << std::endl;
}
};
using func = void(*)();
int main(int argc, const char * argv[])
{
B b;
A *p = (A*)&b;
p->f1();
p->f2();
return 0;
}
这段代码运行为什么输出
A::f1
B::f1
这个你要理解了虚函数表及其内存结构就明白了。大概是这样
A p = (A)&b; 这只是一个地址转换,现在p指向了对象b的地址,并宣称自己的结构是A
p->f1(); 因为A类中f1是非虚函数,所以直接调用A类的f1函数,这个函数地址不在对象上,而是和类相关的,A类的f1自然就输出A::f1
p->f2(); f2是一个虚函数,而虚函数的地址是存放在虚表中,而虚表的地址是在对象上的,所以虽然这时p说自己是A类的指针,但实际指向的虚表还是B,现在调用A类的虚函数f2,f2是A的第一个虚函数,于是就去虚表找第一项的那个函数了,而这个函数实际是B类的第一个函数就是 B的f1了。
参考一下 https://zhuanlan.zhihu.com/p/75172640,有图,讲的还可以
因为基类中f1函数不是虚函数啊,就不会有多态效果