分享一个之前学的知识点,感觉还挺重要的,就是当一个类中的某个数据成员同时拥有就地初始化、构造函数初始化列表和构造函数函数体里的赋值,那么它会先执行哪个?最后生效的又是哪个呢?
根据老师的讲解,数据成员的初始化次序依次为:
就地初始化 > 构造函数的初始化列表 >构造函数里的赋值(严格意义上不能成为初始化)
而当三种初始化方式都有时,构造函的函数体里的赋值肯定执行,并且生效,但是就地初始化和构造函数初始化列表的执行情况是怎样呢?写段代码测试一下

#include<iostream>
#include<cstdio>

using namespace std;

int n = 0;
class STU{
private:
    int id = ++n;       //就地初始化 
public:
    STU(){};         
    STU(int id):id(id){ //初始化列表 
    }
    int getId(){
        return id;
    }
    ~STU(){
    }
};

int main(){
    cout << n << endl;      
    STU s1{};   //调用无参构造
    cout << "n = " << n << ",id = " << s1.getId()<< endl;   
    STU s2{10}; 
    cout << "n = " << n << ",id = " << s2.getId()<< endl;

    return 0;
} 

运行结果为:
0
n = 1,id = 1
n = 1,id = 10
可以看出,当调用无参构造时,id执行了就地初始化,而当调有参构造函数时,id没有执行就地初始化,而是直接执行了构造函数初始化列表。
所以当一个数据成员同时拥有就地初始化和初始化列表时,它会忽略就地初始化而执行构造函数初始化列表。
如果到代码中的有参构造函数的函数体中加上 this->id = 20; ,运行结果会变为:
0
n = 1,id = 1
n = 1,id = 20
可以看到赋值把初始化列表给id初始化的值覆盖掉了,这里在情理之中。

最后修改:2019 年 09 月 21 日
如果觉得我的文章对你有用,请随意赞赏