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

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

相关推荐
wunaiqiezixin1 天前
如何在C++中创建和管理线程
c++
雪度娃娃1 天前
转向现代C++——在意为改写的函数添加 override
开发语言·c++
王老师青少年编程1 天前
csp信奥赛C++高频考点专项训练之前缀和&差分 --【一维差分】:[NOIP 2018 提高组] 铺设道路
c++·前缀和·差分·csp·高频考点·信奥赛·铺设道路
星马梦缘1 天前
aaaaa
数据结构·c++·算法
喵星人工作室1 天前
C++火影忍者1.1.2
开发语言·c++
basketball6161 天前
C++ 中的 ptrdiff_t 详解
开发语言·c++
wunaiqiezixin1 天前
互斥锁与自旋锁的区别
c++
代码中介商1 天前
深入解析STL中的stack、queue与priority_queue
开发语言·c++
磊 子1 天前
STL无序关联容器—unorded_set+unorded_map
开发语言·c++
初夏睡觉1 天前
数据结构学习之~二叉堆 (P3378 【模版】堆)
数据结构·c++·学习