父类构造函数必须在初始化列表中调用,不能在子类构造函数体内调用;多层继承时每层只调用直接父类,虚基类由最派生类显式调用;参数须为常量或构造函数参数,不可依赖未初始化成员;父类构造失败则整个对象构造中止。

父类构造函数必须在初始化列表里调用
子类对象构造时,父类部分必须先完成初始化,而 C++ 规定:基类构造函数**不能**在子类构造函数体内部调用(比如写成 Base::Base(42) 是非法的),只能通过成员初始化列表显式调用。
常见错误是试图在构造函数体内“手动调用”父类构造,结果编译失败,报错类似:error: constructor call in member initializer list is the only way to invoke a base class constructor。
- 正确写法:在子类构造函数声明后、函数体前的冒号
:后列出Base(args) - 如果父类有无参构造函数,且你不显式调用,编译器会自动插入默认调用;但一旦父类没有无参构造,就必须显式写
- 初始化列表中,基类调用必须出现在所有成员变量初始化之前(顺序由继承声明顺序决定,不是书写顺序)
多层继承时怎么指定调用哪一层的父类构造
每层子类只负责调用**直接父类**的构造函数,不跨层。编译器会自动链式展开:最派生类 → 直接基类 → 间接基类。
例如 Derived : public Middle,Middle : public Base,那么 Derived 的初始化列表只需写 Middle(x),而 Middle 自己的初始化列表里写 Base(y) —— Derived 不能也不该直接写 Base(z)。
立即学习“C++免费学习笔记(深入)”;
- 若想让
Derived控制Base的参数,需通过Middle构造函数透传(比如加一个带int base_val的Middle构造重载) - 虚继承下的基类构造由**最派生类**直接负责调用,这点例外 —— 如果
Base是虚基类,Derived初始化列表中必须显式调用Base(...),否则即使Middle调用了也无效
初始化列表里调用父类构造时参数来源受限
初始化列表中所有表达式必须是常量表达式、参数名、或对 this 成员的简单访问(不能调用成员函数,也不能依赖尚未构造完成的成员)。
典型陷阱:想用子类成员变量的值去初始化父类,比如:
class Derived : public Base {
int x = 10;
public:
Derived() : Base(x) {} // ❌ 错误!x 此时尚未初始化
};
- 允许:直接使用构造函数参数,如
Derived(int v) : Base(v) {} - 允许:使用字面量、全局常量、
constexpr函数结果 - 不允许:调用非
constexpr成员函数、访问未初始化的成员、使用this->xxx(此时this已存在但对象未完成构造)
父类构造失败会导致整个对象构造中止
如果父类构造函数抛出异常,子类构造函数体不会执行,且子类的析构函数也不会被调用(因为对象没构造成功)。这是 RAII 安全的关键前提。
这意味着:别指望在子类构造函数体里“兜底处理”父类初始化失败;异常必须向上抛出或在调用点捕获。
- 若父类构造可能失败(如打开文件、分配资源),子类应做好异常传播准备
- 不要在初始化列表中做可能失败的复杂逻辑 —— 应该封装进父类构造,或改用工厂函数 +
std::optional/std::expected(C++23)等更明确的错误传递方式 - 注意:初始化列表中多个基类/成员的构造顺序是固定的(先基类后成员,同级按声明顺序),异常发生时已成功构造的部分会逆序析构










