C++初始化列表

本文主要说明成员初始化列表的注意事项。

I、上帝视角看初始化列表

构造函数可以有两种构造形式,一是在构造函数体内对成员进行赋值,二是使用初始化列表

class Test {
public:
    Test() = default;
    /*
    Test(int x) {           //在函数体内完成
        Test_int = x;
    }
    */

    Test(int x) : Test_int(x) {}    //以初始化列表形式完成

private:
    int Test_int = 0;
};

II、构造函数的两个阶段

构造函数执行分为两个阶段,初始化阶段计算阶段

2.1 初始化阶段

所有类类型的成员都在初始化阶段初始化,即使该成员没有出现在构造函数的初始化列表中。

2.2 计算阶段

用于执行函数体内的赋值操作。

要理解这两个阶段可以观察下面的实例:

#include<iostream>
using namespace std;


class Test1 {
public:
    Test1() {cout << "Construct Test1" << endl;}

    //拷贝构造函数
    Test1(const Test1& other) {
        Test1_int = other.Test1_int;
        cout << "Copy constructor Test1" << endl;
    }

    //拷贝赋值运算符
    Test1& operator=(const Test1& other) {
        //判断自赋值问题,执行swap
        if (this != &other) {
            Test1 temp(other);  //执行一次拷贝构造函数,输出"Copy constructor Test1"

            swap(temp.Test1_int, Test1_int);
        }
        cout << "assignment Test1" << endl;
        return *this;
    }

private:
    int Test1_int = 0;

};

class Test2 {
public:
    //自定义构造函数
    Test2(const Test1& t1) {
        cout << "Construct Test2" << endl;
        test1 = t1;     //执行计算操作
    }
private:
    Test1 test1;
};

int main() {
    Test1 t1;      //输出"Construct Test1"

    cout << "===============================" << endl;

    Test2 t2(t1);   //首先构造Test2中的t1(输出"Construct Test1")
                    //然后调用Test2的自定义构造函数,其中需要调用Test1的拷贝赋值运算符(输出"assignment ...")
                    //在调用拷贝赋值运算符的时候,会调用拷贝构造函数(输出Copy"")
}

执行结果说明:

输出结果

III、初始化列表优势

使用初始化列表主要是考虑性能问题,对于内置类型,如int,float等,使用初始化列表和构造函数体内初始化差别不大,但对于类类型来说,最好使用初始化列表。

看下面例子:

#include<iostream>
using namespace std;


class Test1 {
public:
    Test1() {cout << "Construct Test1" << endl;}

    //拷贝构造函数
    Test1(const Test1& other) {
        Test1_int = other.Test1_int;
        cout << "Copy constructor Test1" << endl;
    }

    //拷贝赋值运算符
    Test1& operator=(const Test1& other) {
        //判断自赋值问题,执行swap
        if (this != &other) {
            Test1 temp(other);  //执行一次拷贝构造函数,输出"Copy constructor Test1"

            swap(temp.Test1_int, Test1_int);
        }
        cout << "assignment Test1" << endl;
        return *this;
    }

private:
    int Test1_int = 0;

};

class Test2 {
public:
    //自定义构造函数
    Test2(const Test1& t1) {
        cout << "Construct Test2" << endl;
        test1 = t1;     //执行计算操作
    }
private:
    Test1 test1;
};

class Test3 {
public:
    Test3(const Test1& t1):test1(t1){
        cout << "Construct Test3" << endl;
    }
private:
    Test1 test1;
};

int main() {
    Test1 t1;      //输出"Construct Test1"

    cout << "==============Test2=================" << endl;

    Test2 t2(t1);   //首先构造Test2中的t1(输出"Construct Test1")
                    //然后调用Test2的自定义构造函数,其中需要调用Test1的拷贝赋值运算符(输出"assignment ...")
                    //在调用拷贝赋值运算符的时候,会调用拷贝构造函数(输出Copy"")

    cout << "==============Test3==================" << endl;
    Test3 t3(t1);

    return 0;
}

执行结果说明:

执行结果

可以看出相比于Test2的函数体内初始化,Test3使用初始化列表可以少调用Test1的构造函数。这对于数据密集型的类来说,是非常高效的。

IV、必须使用初始化列表的情况

除了III中说明的性能原因之外,有些情况下初始化列表是不可或缺的:
1、常量成员引用类型成员:这两种成员变量只能在定义时初始化,而不能重写赋值,所以要写在初始化列表中。
2、没有默认构造函数的类类型成员。由上面的例子可以看出,使用初始化列表不用调用类类型成员的默认构造函数来初始化,而是直接使用拷贝构造函数初始化。

如下面例子:

#include<iostream>

using namespace std;

class Test1 {
public:
    Test1(int a) :Test1_int(a) {}

private:
    int Test1_int;
};

class Test2 {
public:
    Test2(const Test1& t1) {
        test1 = t1;
    }
private:
    Test1 test1;
};

int main() {
    Test1 t1;
    Test2 t2(t1);

    return 0;
}
//无法完成编译

上述代码无法完成编译,因为Test2使用函数体内完成初始化,需要调用Test1的默认初始化函数,而Test1中没有默认初始化函数。

为Test1中增加默认初始化函数,则可完成Test2的初始化:

#include<iostream>

using namespace std;

class Test1 {
public:
    Test1(int a):Test1_int(a) {}
    Test1() = default;

private:
    int Test1_int;
};

class Test2 {
public:
    Test2(const Test1& t1) {
        test1 = t1;
    }
private:
    Test1 test1;
};

int main() {
    Test1 t1;
    Test2 t2(t1);

    return 0;
}

V 、初始化列表中初始化成员的顺序

初始化列表中初始化成员的顺序是按照成员变量声明的顺序完成的,而不是以初始化列表中的顺序。

例如,下列示例:

#include<iostream>

using namespace std;

class Test1 {
public:
    Test1(int a):j(a), i(j) {}    //i未定义

    int i;
    int j;
};

int main() {
    Test1 t1(100);

    cout << t1.i << " " << t1.j << endl;

    return 0;
}

输出结果为:

输出结果

这是由于i未定义的原因。

【参考】
[1] C++ 初始化列表

欢迎转载,转载请注明出处wenmingxing C++ 初始化列表

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343