只有通过基类指针或者引用调用虚函数才能引发动态绑定
虚函数不能声明为静态的,也不能是友元函数,只能是成员函数 因为静态函数没有this指针,因为静态函数是类共享的,就不是对象的一部分,就没有办法逐对象的头4个字节vptr虚表指针,进而也没办法找到虚表,所以静态函数不能声明为虚的
eg:P34\01.cpp
//演示动态绑定的语法的eg #include <iostream> using namespace std; class Base { public: virtual void Fun1() { cout<<"Base::Fun1 ..."<<endl; } virtual void Fun2() { cout<<"Base::Fun2 ..."<<endl; } void Fun3() { cout<<"Base::Fun3 ..."<<endl; } }; class Derived : public Base { public: //void Fun1()没有virtual关键字,Fun1也是虚函数,Fun2类似 virtual void Fun1() { cout<<"Derived::Fun1 ..."<<endl; } virtual void Fun2() { cout<<"Derived::Fun2 ..."<<endl; } void Fun3() { cout<<"Derived::Fun3 ..."<<endl; } }; int main(void) { //基类指针指向派生类对象 Base* p; Derived d; p = &d; //调用的是基类的Fun1还是派生类的Fun1是在运行期时决定的 p->Fun1();//Fun1是虚函数,基类的指针指向派生类对象,调用的是派生类对象的虚函数 p->Fun2(); p->Fun3();//Fun3是非虚函数,根据p指针实际类型来调用相应类的成员函数 return 0; }测试: 基类指针指向派生类对象,调用的是虚函数时,调用实际所指向的虚函数; 如果是普通的函数,会依据指针类型来确定调用的函数;
eg:P34\02.cpp
#include <iostream> using namespace std; class Base { public: virtual void Fun1() { cout<<"Base::Fun1 ..."<<endl; } virtual void Fun2() { cout<<"Base::Fun2 ..."<<endl; } void Fun3() { cout<<"Base::Fun3 ..."<<endl; } Base() { cout<<"Base ..."<<endl; } ~Base() { cout<<"~Base ..."<<endl; } }; class Derived : public Base { public: //void Fun1()没有virtual关键字,Fun1也是虚函数,Fun2类似 /*virtual*/ void Fun1() { cout<<"Derived::Fun1 ..."<<endl; } /*virtual*/ void Fun2() { cout<<"Derived::Fun2 ..."<<endl; } void Fun3() { cout<<"Derived::Fun3 ..."<<endl; } Derived() { cout<<"Derived ..."<<endl; } ~Derived() { cout<<"~Derived ..."<<endl; } }; int main(void) { //基类指针指向派生类对象 Base* p; p = new Derived;//先构造基类,再构造派生类 p->Fun1(); //这里派生类的析构函数不会调用,是因为它认为释放的类型是Base*类型,所以只会调用基类的析构函数,因为析构函数不是虚的; //若析构函数是虚的,它认为p所指向的类型是派生类,也就是说p是一个派生类的对象,派生类的释放会先调用基类的析构函数,再调用派生类的析构函数; delete p; return 0; }测试: 析构函数也可以虚的eg 这里派生类的析构函数不会调用,是因为它认为释放的类型是Base*类型,所以只会调用基类的析构函数,因为析构函数不是虚的; 若析构函数是虚的,它认为p所指向的类型是派生类,也就是说p是一个派生类的对象,派生类的释放会先调用基类的析构函数,再调用派生类的析构函数;
何时需要虚析构函数?
当你可能通过基类指针删除派生类对象时,如果没有将基类的析构函数定义为虚析构函数,那么它只会调用基类的析构函数,而不会调用派生类的析构函数,这样就有可能存在内存泄漏的风险
如果你打算允许其他人通过基类指针调用对象的析构函数(通过delete这样做是正常的),并且被析构的对象是有重要的析构函数的派生类的对象,就需要让基类的析构函数作为虚函数。
P34\02.cpp
#include <iostream> using namespace std; class Base { public: virtual void Fun1() { cout<<"Base::Fun1 ..."<<endl; } virtual void Fun2() { cout<<"Base::Fun2 ..."<<endl; } void Fun3() { cout<<"Base::Fun3 ..."<<endl; } Base() { cout<<"Base ..."<<endl; } //什么时候用到虚析构函数? //如果一个类要做为多态基类,要将析构函数定义为虚函数,防止内存泄漏 virtual ~Base() { cout<<"~Base ..."<<endl; } }; class Derived : public Base { public: //void Fun1()没有virtual关键字,Fun1也是虚函数,Fun2类似 /*virtual*/ void Fun1() { cout<<"Derived::Fun1 ..."<<endl; } /*virtual*/ void Fun2() { cout<<"Derived::Fun2 ..."<<endl; } void Fun3() { cout<<"Derived::Fun3 ..."<<endl; } Derived() { cout<<"Derived ..."<<endl; } //基类的析构函数四虚函数,派生类的自然也变成虚的 ~Derived() { cout<<"~Derived ..."<<endl; } }; int main(void) { //基类指针指向派生类对象 Base* p; p = new Derived;//先构造基类,再构造派生类 p->Fun1(); delete p; return 0; } 测试:虚函数的动态绑定是通过虚表来实现的
包含虚函数的类的头4个字节存放指向虚表的指针,即:虚表指针指向虚表
eg:P34\03.cpp
#include <iostream> using namespace std; class Base { public: virtual void Fun1() { cout<<"Base::Fun1 ..."<<endl; } virtual void Fun2() { cout<<"Base::Fun2 ..."<<endl; } int data1_; }; //派生类有3个虚函数 class Derived : public Base { //会覆盖基类的虚函数Fun2() void Fun2() { cout<<"Derived::Fun2 ..."<<endl; } virtual void Fun3() { cout<<"Derived::Fun3 ..."<<endl; } int data2_; }; //使用typedef来定义一个类型 typedef void (*FUNC)(); int main(void) { cout<<sizeof(Base)<<endl; cout<<sizeof(Derived)<<endl; Base b; long** p = (long**)&b;//p[0][0]指向基类的虚函数Base::Func1 FUNC fun = (FUNC)p[0][0];//将p[0][0]指针强制转换为func,强制转换为函数指针类型 fun();//会调用到基类的虚函数 Derived d; p = long(**)&d; fun = (FUNC)p[0][0]; fun(); fun = (FUNC)p[0][1]; fun(); fun = (FUNC)p[0][2]; fun(); //只有通过基类指针或者基类引用,才有动态绑定 //基类指针pp指向派生类对象,会取出对象d的头4个字节,指向虚表,然后从虚表中找到对应的Func2,实施动态绑定 //调用Func2进行偏移,偏移到Derived::Fun2,而不是Base::Fun2,结合派生类内存图去看 //pp虽然是基类指针类型,但是调用的是派生类的虚函数 Base* pp = &d; pp->Fun2();//Func2是虚函数,运行时期才会去确定Func2的入口地址 //虚函数可以直接使用,但不是动态绑定,入口地址在编译期就决定了 //直接调用是静态绑定 d.fun2(); return 0; }分析 (1)虚表与虚基类表不一样,虚基类表再虚继承的时候会产生 基类与派生类的内存模型 (2)基类大小为什么是8?因为头四个字节存放的是虚表指针(指向虚表的指针),4个字节,所以4(基类只有一个数据成员)+4=8 (3)基类的内存模型,如下: Base类中第一个位置存放的是指针,所以地址是指针的指针,存放的是指针,该指针指向了一张表格; 虚表vtbl中存放的是函数指针; 虚表里面有2个虚函数 (4)派生类的内存模型,如下: 虚表里面有3个虚函数
测试,验证如下:
该eg源自MFC框架
测试: