//多态的原理--虚函数指针--子类虚函数指针初始化 #include<iostream> using namespace std; /* 多态的实现原理(有自己猜想部分) 基础知识: 类中的成员函数本质上是C语言中的全局函数,只是在全局函数的参数列表中多加了一个结构体指针参数 详解: 对于类中没有用virtual关键字修饰的成员函数,c++编译器在静态编译的时候,c++就会确定对象调用的全局函数 当类中声明虚函数时,c++编译器会在静态编译的时候为这个类生成一个虚函数表, 虚函数表是一个存储类成员函数指针的数据结构, 一个虚函数表只属于一个类 虚函数表是由编译器自动生成与维护的 virtual成员函数的地址会被c++编译器放入虚函数表中 在定义一个对象的时候即运行时或者说动态编译的时候(未调用构造函数之前)---Point p1; , 那么c++编译器会为这个对象 隐式的 分配4个字节大小的内存, 这个内存里是一个 指针变量 此时这个指针变量还是为NULL,当执行函数的构造函数的时候 c++编译器会默认的为这个指针变量赋值 这个指针变量会指向该类的虚函数表 对于子类而言,子类的初始化比较特殊,必须先调用父类的构造函数,这时候这个隐藏的指针变量会被初始化为父类中虚函数表的地址 随后子类对象又会再次调用自身的构造函数 这个隐藏的指针变量又会再一次被赋值为 子类 对应的类的虚函数表的地址 说明1: 通过虚函数表指针VPTR调用重写函数是在程序运行时进行的,因此需要通过寻址操作才能确定真正应该调用的函数。 而普通成员函数是在编译时就确定了调用的函数。在效率上,虚函数的效率要低很多。 说明2: 出于效率考虑,没有必要将所有成员函数都声明为虚函数 */ class Point{ public: Point(){ PrintA(); } virtual void PrintA(){ cout << "1 我是第一个父类虚函数 我必将产生占据4个字节大小的函数指针a " << endl; } virtual void PrintB(){ cout << "2 我是第二个父类虚函数 我必将产生占据4个字节大小的函数指针b " << endl; } virtual void PrintC(){ cout << "3 我是第三个父类虚函数 我必将产生占据4个字节大小的函数指针c " << endl; } private: int b; }; class PointA :public Point{ public: void PrintA(){ cout << "我是子类PointA 我重写了父类的虚函数 " << endl; } }; class PointB :public PointA{ void PrintA(){ cout << "我是孙子类 PointB 我重写了字类的重写函数 用来验证子类中的重写函数是不是一个虚函数 " << endl; } }; void ProtectA(PointA &pin){ pin.PrintA(); } void ProtectB(){ Point p1; cout << "Point类型的大小" << sizeof(p1) << endl; //打印 8 //说明:c++编译器的确为虚函数分配了4个字节大小的内存 并且无论有多少个虚函数,只会分配4个字节大小的内存空间 //侧面证明了 虚函数指针指向的是一个虚函数表 而不是一个虚函数指针指向一个虚函数 PointA pa; cout << "PointA类型的大小" << sizeof(pa) << endl; //打印 8 //根据结果说明:子类重写父类的虚函数,虽然没有加virtual关键字,但是本质上还是一个虚函数 //不然 为什么c++编译器为什么会为子类对象多分配4个字节大小的内存空间呢? //做一下验证 写一个子类PointB 继承 PointA看是否能实现多态 PointB pb; ProtectA(pb);// 打印出PointB 孙子类中的重写函数 //证明: 子类重写父类的虚函数,虽然没有加virtual关键字,但是本质上还是一个虚函数 } void ProtectC(){ //验证子类的分步初始化对虚函数指针的影响----我在父类的构造函数中调用一个虚函数,并且在子类中重写该虚函数 PointA pa;//调用了父类的PrintA()函数 //验证我文章开头结论 pa.PrintB(); } void main(){ ProtectC(); system("pause"); }