0629第五讲继承(3)虚函数、多态

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个内存),这个类的虚函数表包含一个数组用来存放虚函数的地址,每一个指针指向了类中的虚函数地址。
当虚函数被调用时,编译器会使用该对象中的虚指针来查找虚函数表,然后遍历虚函数表(使得程序运行速度减慢),以查找到虚函数的指针(地址),最终找到正确版本的函数。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值