Fellow_ 2023-03-25 19:59 采纳率: 100%
浏览 42
已结题

派生类通过作用域运算符调用基类虚函数是怎么做到的?

派生类重写了基类的虚函数,派生类对象通过作用域运算符成功调用了基类的虚函数。重写虚函数不是已经覆盖了虚函数表中基类虚函数的地址吗,为什么还能调用基类的虚函数?

代码如下:

class base
{
public:
    base()
    {
        cout << "base()" << endl;
    }
    virtual void func()
    {
        cout << "base::func()" << endl;
    }
};

class sub:public base
{
public:
    sub() :base()
    {
        cout << "sub()" << endl;
    }

    void func()
    {
        cout << "sub::func()" << endl;
    }
};

int main() {
    sub s;
    base& b = s;

    b.func();
    s.base::func();
}

运行

img

  • 写回答

3条回答 默认 最新

  • 海洋 之心 2022年度博客之星人工智能领域TOP 1 2023-03-25 20:07
    关注
    • 派生类重写基类的虚函数时,虽然会覆盖虚函数表中基类虚函数的地址,但是基类的虚函数仍然存在于派生类的对象中,可以通过作用域运算符来访问它。因此,当派生类对象通过作用域运算符来调用基类的虚函数时,实际上是直接访问了派生类对象中的基类虚函数地址,而不是通过虚函数表来调用。

    • 具体来说,派生类对象在创建时会包含基类对象的所有成员,包括基类的虚函数表。当派生类重写基类的虚函数时,它会在自己的虚函数表中填入新的函数地址,覆盖基类虚函数表中相应的函数地址。但是,派生类对象中仍然包含了基类的虚函数表,其中仍然保存着基类的虚函数地址。因此,当派生类对象通过作用域运算符来访问基类的虚函数时,它实际上是直接访问派生类对象中的基类虚函数地址,而不是通过虚函数表来调用。

    • 需要注意的是,当基类的虚函数被重写时,为了避免出现未定义的行为,应当将基类的虚函数声明为虚函数,并在派生类中使用override关键字来覆盖它。这样可以确保在派生类对象上调用基类虚函数时,可以正确地访问到派生类对象中的基类虚函数地址。

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(2条)

报告相同问题?

问题事件

  • 已结题 (查看结题原因) 3月26日
  • 已采纳回答 3月25日
  • 创建了问题 3月25日

悬赏问题

  • ¥15 echarts动画效果失效的问题。官网下载的例子。
  • ¥60 许可证msc licensing软件报错显示已有相同版本软件,但是下一步显示无法读取日志目录。
  • ¥15 Attention is all you need 的代码运行
  • ¥15 一个服务器已经有一个系统了如果用usb再装一个系统,原来的系统会被覆盖掉吗
  • ¥15 使用esm_msa1_t12_100M_UR50S蛋白质语言模型进行零样本预测时,终端显示出了sequence handled的进度条,但是并不出结果就自动终止回到命令提示行了是怎么回事:
  • ¥15 前置放大电路与功率放大电路相连放大倍数出现问题
  • ¥30 关于<main>标签页面跳转的问题
  • ¥80 部署运行web自动化项目
  • ¥15 腾讯云如何建立同一个项目中物模型之间的联系
  • ¥30 VMware 云桌面水印如何添加