[温习C/C++]0x09 C++构造函数中调用虚函数会发生什么?

系列文章目录

[温习C/C++]0x00-STL标准模板库概述
[温习C/C++]0x01-STL泛型算法-持续更新长文版
[温习C/C++]0x03-sort排序
[温习C/C++]0x04 C++刷题基础编码技巧
[温习C/C++]0x05 C++刷题技巧---set自定义排序及查找
[温习C/C++]0x06 坐标系中矩形重叠类问题分析
[温习C/C++]0x07 C++刷题技巧---字符串查找find、find_if、find_first_of和find_last_of
[温习C/C++]0x08 C++刷题技巧---关联容器使用operator[]访问避坑
[温习C/C++]0x09 C++构造函数中调用虚函数会发生什么?

C++构造函数中调用虚函数会发生什么?

构造函数中调用虚函数案例

cpp 复制代码
#include <iostream>  
  
class B {  
public:  
    B() {  
        f(); // 关键点:构造函数中调用虚函数  
    }  
    virtual void f() {  
        std::cout << "base f()" << std::endl;  
    }  
};  
  
// C 必须公有继承 B,才能体现虚函数重写  
class C : public B {  
public:  
    C() {  
        f();  
    }  
    // 重写基类的虚函数  
    void f() override {  
        std::cout << "derive f()" << std::endl;  
    }  
};  
  
int main() {
	// B* obj = reinterpret_cast<B *>(new C());
    B* obj = new C();  
    obj->f();  
    return 0;  
}
  • 输出
bash 复制代码
base f()
derive f()
derive f()

执行流程分析

  1. 创建C对象 : new C()

    • 首先调用基类B的构造函数
    • 在B的构造函数中调用 f() ,此时调用的是 B::f() (基类版本),因为C的部分还没有被构造
    • 然后调用C的构造函数
    • 在C的构造函数中调用 f() ,此时调用的是 C::f() (派生类版本),因为C的部分已经开始构造
  2. 通过B*指针调用f() : obj->f()

    • 此时对象已经完全构造完成,虚函数表已经正确设置,所以会调用 C::f() (派生类版本)

技术要点

虚函数在构造函数中的行为

  • 构造函数中调用虚函数不会触发多态 :当在基类构造函数中调用虚函数时,只会调用基类自己的版本,而不是派生类的重写版本。

  • 原因 :在构造基类时,派生类的部分还没有被构造,虚函数表还没有完全设置好,所以无法调用派生类的重写函数。

类的声明顺序

  • 在C++中,类必须在使用前声明。如果类C继承自类B,那么类B必须在类C之前声明。

类型转换

  • reinterpret_cast<B >(new C()) 是不必要的,因为C公有继承自B,所以可以直接隐式转换: B obj = new C();

总结

C++中一个重要的特性:

构造函数中调用虚函数不会发生多态 。
这是因为在构造过程中,对象的类型是从基类到派生类逐渐构建的,在基类构造函数执行时,派生类的部分还不存在,所以虚函数调用会被解析为基类的版本。

相关推荐
三品吉他手会点灯1 小时前
STM32 VSCode 开发-C/C++的环境配置中,找不到C/C++: Edit Configurations选项
c语言·c++·vscode·stm32·单片机·嵌入式硬件·编辑器
6Hzlia2 小时前
【Hot 100 刷题计划】 LeetCode 287. 寻找重复数 | C++ 数组判环 (快慢指针终极解法)
c++·算法·leetcode
Robot_Nav3 小时前
DPMPC-Planner:复杂静态环境与动态障碍物下的无人机实时轨迹规划框架
c++·无人机·mpc
水饺编程5 小时前
第5章,[标签 Win32] :获取设备环境的信息
c语言·c++·windows·visual studio
lhbian5 小时前
C++、C与易语言:编程语言对比解析
c语言·开发语言·c++
hehelm5 小时前
二叉搜索树
c++
云泽8085 小时前
笔试算法 - 双指针篇(一):移动零、复写零、快乐数与盛水容器
c++·算法
小堃学编程5 小时前
【项目实战】基于protobuf的发布订阅式消息队列(4)—— 服务端
c语言·c++·vscode·消息队列·gtest·protobuf·muduo
小白学大数据5 小时前
解决 Python 爬虫被限制:延迟抓取指令深度解析
开发语言·c++·爬虫·python
会编程的土豆6 小时前
【复习】二分查找
数据结构·c++·算法