在 C++ 中,类的非静态成员变量有三种主要的初始化方式,它们在语法、适用场景和执行顺序上各有特点。以下是清晰总结:
✅ 1.成员初始化列表(Member Initializer List)
最推荐、最高效的方式,尤其适用于:
const成员- 引用成员
- 没有默认构造函数的类类型成员
- 需要避免“先默认构造再赋值”的性能浪费
classStudent{constintid;std::string name;doublescore;public:// 成员初始化列表(构造函数参数后 : 开始)Student(inti,conststd::string&n,doubles):id(i),name(n),score(s)// 直接初始化,非赋值!{}};🔹优点:
- 对象直接构造为目标值,无临时对象或二次赋值;
- 是初始化
const/引用成员的唯一合法方式。
✅ 2.类内默认成员初始化器(In-Class Member Initializers)
C++11 起支持,在类定义中直接给成员赋初值。
classStudent{intid=0;// 整型默认为 0std::string name="N/A";// 字符串默认为 "N/A"doublescore{};// 值初始化(等价于 0.0)public:Student()=default;// 使用默认值Student(inti,conststd::string&n):id(i),name(n){}// 此时覆盖默认值};🔹规则:
- 若构造函数的初始化列表未指定该成员,则使用类内默认值;
- 若初始化列表显式指定了该成员,则类内默认值被忽略;
- 不能用于
static成员(除非是constexpr)。
⚠️ 3.构造函数体内赋值(Assignment in Constructor Body)
不推荐作为“初始化”手段,本质是“先构造 + 再赋值”。
classStudent{std::string name;public:Student(conststd::string&n){name=n;// ❌ 先调用 string 默认构造,再 operator= 赋值}};🔸问题:
- 对于复杂对象(如
std::string,std::vector),会多一次默认构造 + 一次赋值,效率低;- 无法初始化
const或引用成员(编译错误);- 语义上属于“修改”而非“初始化”。
✅仅适合:需要根据复杂逻辑计算后再赋值的普通成员(但仍建议优先用初始化列表)。
📊 三种方式对比表
| 初始化方式 | 语法位置 | 是否真正“初始化” | 支持 const/引用 | 效率 | 推荐度 |
|---|---|---|---|---|---|
| 成员初始化列表 | 构造函数声明后:后 | ✅ 是 | ✅ 支持 | ⭐⭐⭐ 最高 | ★★★★★ |
| 类内默认初始化器 | 类定义中成员声明处 | ✅ 是 | ✅ 支持 | ⭐⭐⭐ 高 | ★★★★☆ |
| 构造函数体内赋值 | 构造函数{}体内 | ❌ 否(是赋值) | ❌ 不支持 | ⭐ 低 | ★★☆☆☆ |
💡 最佳实践建议
- 优先使用成员初始化列表—— 尤其对资源型对象;
- 用类内默认值提供安全 fallback—— 避免未初始化状态;
- 永远不要在构造函数体内“初始化” const 或引用成员;
- 避免混合使用:若已在初始化列表中初始化,就不要再在函数体内赋值。
🌟记住:
“初始化”发生在对象诞生那一刻,而构造函数体内的代码是在对象已经存在之后才执行的。
所以,真正的初始化,只发生在初始化列表和类内默认值中。