初始化列表是一种C++初始化列表,有初始化阶段和计算阶段两个阶段。
与其他函数不同,构造函数除了有名字,参数列表和函数体之外,还可以有初始化列表,初始化列表以冒号开头,后跟一系列以逗号分隔的初始化字段。
从概念上来讲,构造函数的执行可以分成两个阶段,初始化阶段和计算阶段,初始化阶段先于计算阶段。
所有类类型(class type)的成员都会在初始化阶段初始化,即使该成员没有出现在构造函数的初始化列表中
一般用于执行构造函数体内的赋值操作。
与其他函数不同,构造函数除了有名字,参数列表和函数体之外,还可以有初始化列表,初始化列表以冒号开头,后跟一系列以逗号分隔的初始化字段。
class foo { public: foo(string s, int i) :name(s), id(i) {}; // 初始化列表 private: string name; int id; };
下面的代码定义两个结构体,其中Test1有构造函数,拷贝构造函数及赋值运算符,为的是方便查看结果。Test2是个测试类,它以Test1的对象为成员,我们看一下Test2的构造函数是怎么样执行的。
#include <iostream> using namespace std; struct Test1 { Test1() // 无参构造函数 { cout << "Construct Test1" << endl; } Test1(const Test1& t1) // 拷贝构造函数 { cout << "Copy constructor for Test1" << endl; this->a = t1.a; } Test1& operator = (const Test1& t1) // 赋值运算符 { cout << "assignment for Test1" << endl; this->a = t1.a; return *this; } int a; }; struct Test2 { Test1 test1; Test2(Test1& t1) { test1 = t1; } }; int main() { Test1 t1; Test2 t2(t1); } //Construct Test1 //Construct Test1 //assignment for Test1
解释一下,第一行输出对应调用代码中第一行,构造一个Test1对象。第二行输出对应Test2构造函数中的代码,用默认的构造函数初始化对象test1,这就是所谓的初始化阶段。第三行输出对应Test1的赋值运算符,对test1执行赋值操作,这就是所谓的计算阶段。
#include <iostream> using namespace std; struct Test1 { Test1() // 无参构造函数 { cout << "Construct Test1" << endl; } Test1(const Test1& t1) // 拷贝构造函数 { cout << "Copy constructor for Test1" << endl; this->a = t1.a; } Test1& operator = (const Test1& t1) // 赋值运算符 { cout << "assignment for Test1" << endl; this->a = t1.a; return *this; } int a; }; struct Test2 { Test1 test1; Test2(Test1& t1) :test1(t1) { } }; int main() { Test1 t1; Test2 t2(t1); } //Construct Test1 //Copy constructor for Test1
第一行输出对应 调用代码的第一行。第二行输出对应Test2的初始化列表,直接调用拷贝构造函数初始化test1,省去了调用默认构造函数的过程。所以一个好的原则是,能使用初始化列表的时候尽量使用初始化列表。
参考