[温习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++中一个重要的特性:

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

相关推荐
kyle~2 小时前
C++---yaml-cpp YAML标准解析/生成库
c++·参数
96772 小时前
多线程编程:整个互斥的流程以及scoped_lock的用法,以及作用,以及 硬件上的原子操作和逻辑上的原子操作
开发语言·c++·算法
liuyao_xianhui2 小时前
优选算法_topk问题_快速排序算法_堆_C++
java·开发语言·数据结构·c++·算法·链表·排序算法
yunn_2 小时前
Qt智能指针
c++·qt
liuyao_xianhui2 小时前
优选算法_堆_最后一块石头的重量_C++
java·开发语言·c++·算法·链表
上天_去_做颗惺星 EVE_BLUE2 小时前
Linux Core Dump 测试操作手册
linux·c++·测试工具
羊小猪~~2 小时前
算法/力扣--栈与队列经典题目
开发语言·c++·后端·考研·算法·leetcode·职场和发展
云泽8082 小时前
深入红黑树:SGI-STL 中 map 与 set 的关联容器架构剖析
开发语言·c++·stl底层架构
福楠2 小时前
constexpr 全家桶
c语言·开发语言·c++