0629第五讲继承(3)虚函数、多态
1、虚函数
* 1.1虚函数的使用
当父类指针或引用指向子类对象,调用被覆盖的函数,当想调用到正确版本函数,需要把该成员函数声明为虚函数(virtual)
#include <iostream>
using namespace std;
class Base{
public:
virtual void print(){
cout<<"Base class function"<<endl;
}
};
class Derived1:public Base{
public:
void print(){
cout<<"Derived1 class function"<<endl;
}
};
class Derived2:public Base{
public:
void print(){
cout<<"Derived2 class function"<<endl;
}
};
int main(int argc, const char * argv[]) {
Base b1;
Derived1 d1;
Derived2 d2;
b1=d1;//子类对象赋值给父类对象
b1.print();//父类被调用
Base &b11=d1;//父类引用指向子类1
b11.print();//子类1被调用
Base &b12=d2;//父类引用指向子类2
b12.print();//子类2被调用
Base* b21 = &d1;//父类指针指向子类2
b21->print();//子类1被调用
Base* b22 = &d2;//父类指针指向子类2
b22->print();//子类2被调用
return 0;
}
1.2虚析构函数
如果一个类有子类,那么这个父类的析构函数必须是虚函数(虚析构函数)
原因:如果父类的析构不是虚析构,则当(用delete)删除一个指向子类对象的父类指针时,将调用父类版本的析构函数,子类只释放了来自父类的那部分成员变量,而子类自己扩展的成员没有被释放,造成内存泄漏。
显示删除父类指针,引用,防止内存泄漏1.3动态绑定
- 多态:父类指针、引用,统一管理子类对象
- 动态绑定:虚函数被调用的时候,到底调用哪个版本,在编译的时候无法确定,只有在执行时才能确定,称为动态绑定。
绑定似的程序可以照顾到未来增加的代码,比如创建一个新的子类,并在子类中覆盖了父类的虚函数。用之前的父类指针,依然可以正确的调用到新子类里的函数,而无需对旧有代码进行更改。 - 1.4多态的代价
多态:用父类指针或引用,统一操作各种子类对象
多态的代价:类添加虚函数,会自动创建一个虚指针,虚指针指向一个虚函数表(虚函数表存储指针(地址)),表中指针指向了每一个虚函数。
1)增加内存
2)程序运行速度减慢10-20%
为了实现动态绑定,编译器为每一个包含虚函数的类提供一个虚函数表,这个函数表被一个虚指针指向(正增加内存 char * 8个内存),这个类的虚函数表包含一个数组用来存放虚函数的地址,每一个指针指向了类中的虚函数地址。
当虚函数被调用时,编译器会使用该对象中的虚指针来查找虚函数表,然后遍历虚函数表(使得程序运行速度减慢),以查找到虚函数的指针(地址),最终找到正确版本的函数。