C++虚函数依赖、虚函数表、静态绑定与动态绑定、虚函数与动静态绑定、虚析构函数、多态、抽象类

虚函数依赖

1、  实例化对象的时候,需要虚函数能产生地址,存储在vftable中。

2、  对象必须存在。

所以构造函数不能作为虚函数,因为构造函数调用之后才能产生对象。

同时请注意,在构造函数中调用的任何函数都是静态绑定,即使调用虚函数,也不会发生动态绑定。

同时static方法因为不需要依赖一个实例,也不能加virtural。

虚函数表

如果类里面定义了虚函数,那么编译阶段,编译器给这个类型产生一个唯一的vftable虚函数表,虚函数表中主要存储的内容就是RTTI(run time type information,运行时的类型信息,也叫类型字符串,用于说明虚函数表是哪个类产生的)指针信息和虚函数地址。当程序运行时,每一张虚函数表都会加载到内存的.rodata区。

一个类里面定义了虚函数,那么这个类定义的对象,其运行时,内存中开始部分,多存储一个vfptr虚函数指针。指向相应类型的虚函数表。一个类型定义个n个对象指向的都是同一张虚函数表。

一个类里面虚函数的个数,不影响对象内存的大小,影响的是虚函数表的大小。

如果派生类中的方法和从基类继承来的某个方法,返回值、函数参数列表、函数名都先相同,而且继承的方法是virtual虚函数,那么派生类的这个方法,自动处理成虚函数。覆盖住虚函数表的原定义。

静态绑定与动态绑定

静态绑定:编译器根据函数调用的静态类型来决定调用哪个函数实现。静态类型是指在编译时已知的对象类型或指针类型。

#include <iostream>

class Base {
public:
    void show() {
        std::cout << "Base show()" << std::endl;
    }
};

class Derived : public Base {
public:
    void show() {
        std::cout << "Derived show()" << std::endl;
    }
};

int main() {
    Base base;
    Derived derived;
    
    Base* ptr = &derived;  // 使用基类指针指向派生类对象
    
    ptr->show();  // 静态绑定,调用的是基类的show()函数
    
    return 0;
}

动态绑定:动态绑定是指在运行时确定调用哪个函数,而不是在编译时确定。它允许通过基类指针或引用调用派生类中重写的虚函数。

在下面的代码中,Base 类中的 show 函数被声明为虚函数,并在派生类 Derived 中进行了重写。在 main 函数中,通过将派生类对象的地址赋值给基类指针 ptr,实现了多态性。当我们通过 ptr->show() 调用 show 函数时,编译器不是根据指针的类型(Base 类型)来决定调用哪个函数,而是根据指针实际指向的对象类型(Derived 类型,RTTI里说明的类型)来确定调用哪个函数。

#include <iostream>

class Base {
public:
    virtual void show() {
        std::cout << "Base show()" << std::endl;
    }
};

class Derived : public Base {
public:
    void show() override {
        std::cout << "Derived show()" << std::endl;
    }
};

int main() {
    Base base;
    Derived derived;
    
    Base* ptr = &derived;  // 使用基类指针指向派生类对象
    
    ptr->show();  // 动态绑定,根据指针所指向的实际对象类型调用对应的show()函数
    
    return 0;
} 

虚函数与动静态绑定

不是调用虚函数就一定是动态绑定。

静态绑定:用对象本身调用虚函数是静态绑定。

动态绑定:1、必须指针或引用调用,2、调用的是虚函数。指针(可以是基类指针,也可以是派生类指针)或对象有可能指向自身类的对象,也有可能指向派生类对象,最终都是程序运行的时候才会确定调用哪个虚函数表里的虚函数,这就是动态绑定。

虚析构函数

基类的指针或引用指向堆上new出来的派生类对象的时候,delete行为调用析构函数必须发生动态绑定,否则会导致派生类的析构函数无法使用。

在继承关系中,如果基类的析构函数不是虚函数,当通过一个基类指针删除指向派生类对象的对象时,只会调用基类的析构函数,而不会调用派生类的析构函数。这可能导致派生类对象中的资源没有正确释放,造成资源泄漏。

通过将基类的析构函数声明为虚析构函数,可以解决这个问题,因为通过这方法派生类下析构函数。当通过基类指针删除指向派生类对象的对象时,将会调用派生类的析构函数,确保派生类对象中的资源得到正确释放。

我们通过基类指针 ptr 指向派生类对象,并使用 delete 运算符删除对象。由于基类的析构函数是虚析构函数,因此会按照派生类到基类的顺序调用析构函数。

多态

C++中的多态是值同名函数有多种不同的实现方式。

多态分为静态多态和动态多态。

静态多态:编译时期的多态,函数重载,类模板,在编译阶段就确定好的函数版本。

动态多态:在继承结构中,基类指针或引用指向派生类对象,通过该指针调用同名覆盖方法。基类指针指向哪个派生类对象,就会调用哪个派生类对象的同名覆盖方法。

抽象类

拥有纯虚函数的类,叫做抽象类。

定义抽象类的初衷,并不是让它抽象某个实体的类型,而是为了让它的派生类们通过继承可以直接复用相同的属性和方法,保留统一的接口。

抽象类不能实例化对象,但是可以定义指针和引用变量。

  • 9
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值