C++成员变量的初始化顺序

本文介绍了C++中成员变量的初始化顺序仅依赖于它们在类中的声明顺序,而非构造函数初始化列表中的顺序。通过一个具体示例展示了这一特性,并解释了背后的原因。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

    知识点: 在C++中,成员变量的初始化顺序只与它们在类中声明的顺序有关,而与初始化列表中的顺序无关。
    不知道大家知不知道这个,反正我是不知道(或者忘了),我是在《剑指offer》上看到的,后来在《Effective C++》中条款4中再次看到,特地在这记录一下,防止忘记。
    
    下面举个具体的例子,摘自《剑指offer》:
class A
{
private:
    int n1;
    int n2;
public:
    A() : n2(0), n1(n2  + 2)
    {
    }

    void Print()
    {
        std::cout << "n1: " << n1 << ", n2: " << n2 << std::endl;
    }
};

int _tmain(int argc, _TCHAR* argv[])
{
    A a;
    a.Print();

    return 0;
}
    如果你不懂上面的知识点,肯定一开始和我的答案是一样的,n1为2,n2为0。如果运用上面的知识点,正确答案是:n1是随机值,n2是0。所以大家一定要注意了。
    为什么C++要这么做呢,我查了下资料,引自这篇文章( http://www.cnblogs.com/lidabo/p/3790606.html,感谢作者)的一句话: 因为成员变量的初始化次序是根据变量在内存中次序有关,而内存中的排列顺序早在编译期就根据变量的定义次序决定了。
### C++成员变量初始化方式 在 C++ 编程语言中,成员变量可以通过多种方式进行初始化。这些方法主要包括 **构造函数体内的赋值** 和 **初始化列表** 的使用。每种方法都有其特定的应用场景和注意事项。 #### 1. 构造函数体内的赋值 当成员变量在构造函数体内被赋值时,实际上是先进行了默认初始化(如果存在),然后再通过赋值操作符将其修改为目标值。这种方式适合于普通的非 `const` 非引用类型的成员变量[^2]。 ```cpp class MyClass { public: int value; MyClass(int v) { value = v; // 这里是在构造函数体内进行赋值 } }; ``` 然而,对于某些特殊类型的成员变量(如 `const` 类型、引用类型以及没有默认构造函数的对象类型),这种方法并不适用[^3]。 --- #### 2. 初始化列表 初始化列表是一种更高效的方式,在对象创建过程中直接完成成员变量初始化工作。它的优点在于可以处理那些无法通过简单赋值来实现初始化的情况,比如: - 没有默认构造函数的成员变量; - `const` 类型的成员变量; - 引用类型的成员变量。 ##### 使用示例 ```cpp class MyClass { private: const int cValue; std::string& refString; public: MyClass(const int cv, std::string& str) : cValue(cv), // 初始化常量成员变量 refString(str) { // 初始化引用成员变量 } void display() { std::cout << "Constant Value: " << cValue << ", Reference String: " << refString << std::endl; } }; int main() { std::string s = "Example"; MyClass obj(42, s); obj.display(); } ``` 上述代码展示了如何利用初始化列表为 `const` 成员变量和引用成员变量提供初始值[^1]。 --- #### 3. 声明时初始化 (C++11 及之后版本支持) 自 C++11 起,还可以在类定义内部直接为成员变量指定默认值。这种做法简化了一些情况下对初始化列表的需求,但仍然遵循成员变量声明顺序决定的实际初始化次序[^4]。 ```cpp class MyClass { private: int defaultValue = 0; // 声明时初始化 double piValue = 3.14159; // 同样也是声明时初始化 public: MyClass() {} }; ``` 尽管如此,仍需注意的是,即便提供了此类内初始化表达式,一旦某个具体实例化过程涉及不同的初值设定,则优先采用后者覆盖前者[^5]。 --- #### 总结对比 | 特性 | 构造函数体内赋值 | 初始化列表 | 声明时初始化 | |-------------------------|------------------------------------------|----------------------------------------|-----------------------------------| | 是否支持复杂类型 | 不完全支持 | 完全支持 | 支持,默认行为 | | 效率 | 较低 | 更高 | 视具体情况而定 | | 应用范围 | 普通数据类型 | 包括但不限于无默认构造函数、`const`/引用 | 所有可赋予默认值的数据类型 | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值