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

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

相关推荐
博客18002 天前
酷宝的使用方法,超好用的免费界面库,C++、MFC可用
c++·mfc·界面库·库来帮·酷宝
郝学胜_神的一滴2 天前
CMake 026:属性体系精讲、四大作用域全解 & 实战代码落地
c++·cmake
众少成多积小致巨2 天前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
clint4566 天前
C++进阶(1)——前景提要
c++
夜悊7 天前
C++代码示例:进制数简单生成工具
c++
郝学胜_神的一滴7 天前
CMake 021: IF 条件判据详诠
c++·cmake
_wyt0017 天前
洛谷 B3930 [GESP202312 五级] 烹饪问题 题解
c++·gesp
玖玥拾7 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
один but you7 天前
constexpr函数
c++
凡人叶枫7 天前
Effective C++ 条款41:了解隐式接口和编译期多态
java·开发语言·c++·effective c++