构造函数名必须与类名大小写完全一致,无返回类型;必须用初始化列表初始化const/引用/无默认构造函数的成员;定义带参构造函数后默认构造函数不再自动生成;委托构造函数不可循环调用。

构造函数名字必须和类名完全一致
这是最常踩的坑:C++ 构造函数没有返回类型(连 void 都不能写),且函数名必须与类名大小写完全相同。写成 MyClass() 是对的,写成 myclass()、init() 或 void MyClass() 都会编译失败,报错类似:no matching function for call to 'MyClass::MyClass()'。
实操建议:
立即学习“C++免费学习笔记(深入)”;
- 定义类时,立刻在类内声明同名函数,避免后期补漏时拼错
- 如果类名含下划线或大驼峰(如
HttpHandler),构造函数必须严格匹配,不接受任何形式的缩写或风格转换 - IDE 通常能自动补全构造函数签名,但别盲目信任——检查是否真和类名一模一样
初始化列表比赋值更高效且必要
在构造函数体里用 = 给成员赋值(比如 name = "test";)看似自然,但对自定义类型、const 成员或引用成员根本不可行。正确做法是用初始化列表(: 后面那段)。
实操建议:
立即学习“C++免费学习笔记(深入)”;
- 所有
const成员、引用成员(如int& ref;)、无默认构造函数的类类型成员,必须出现在初始化列表中 - 即使不是强制要求,也优先用初始化列表初始化内置类型(如
int x{0};),避免先调默认构造再赋值的冗余开销 - 初始化顺序只取决于成员在类中声明的顺序,跟初始化列表里的书写顺序无关——这点容易误判,调试时注意
class Person {
const std::string name;
int& age_ref;
public:
Person(const std::string& n, int& a) : name(n), age_ref(a) {} // ✅ 必须这样
};默认构造函数会被编译器悄悄删除
只要定义了任意一个带参数的构造函数,编译器就不会再自动生成默认构造函数(MyClass())。这时如果代码里写了 MyClass obj;,就会报错:no matching constructor for initialization of 'MyClass'。
实操建议:
立即学习“C++免费学习笔记(深入)”;
- 如果需要支持无参创建,显式定义一个默认构造函数,哪怕空实现:
MyClass() = default;或MyClass() {} - 用
= default更安全,它保留编译器生成的 trivial 特性,对 POD 类型、聚合初始化等场景有影响 - 如果类有资源管理(如指针、文件句柄),别依赖编译器默认行为——自己写清楚每个构造路径
委托构造函数要小心循环调用
C++11 支持一个构造函数调用另一个构造函数(委托),语法是 : this(args),但它只能出现在初始化列表里,且不能形成调用环。一旦写错,编译器可能不报错但运行时崩溃,或直接拒绝编译。
实操建议:
立即学习“C++免费学习笔记(深入)”;
- 委托目标必须是同一类的其他构造函数,不能跨继承链,也不能委托给自己
- 被委托的构造函数仍会完整执行(包括其初始化列表和函数体),所以别在委托后又重复初始化同一成员
- 调试时如果遇到栈溢出或未定义行为,优先检查构造函数之间是否存在隐式/显式循环委托
class Vec {
int* data;
size_t sz;
public:
Vec() : Vec(10) {} // ✅ 委托到下面
Vec(size_t n) : data(new int[n]), sz(n) {}
Vec(size_t n) : Vec() {} // ❌ 错误:Vec() 又调 Vec(),死循环
};构造函数看着简单,但涉及初始化顺序、委托逻辑、默认行为取消这些细节,稍不注意就掉进静默错误或运行时陷阱里。尤其在类有继承、模板或移动语义时,初始化链条会变得更长,每一步都得明确控制。











