导语
uniform initialization(标准初始化)
在c++11之前的版本,初始化是一个没有被统一的写法,有时候用小括号,有时候用大括号,有时候还用赋值号,在c++11中就统一了初始化的形式——“{ }”,直接在对象名后面跟大括号,并在大括号中写入需要初始化的值,并用逗号隔开。甚至类构造函数的初始化列表也可以用“{}”大括号。
1.就地初始化
1.1 简介
在C++11之前,只能对结构体或类的静态常量成员就行就地初始化,其他的不行。如下代码所示:1
2
3
4
5
6class Test
{
private:
static const int a=10; //yes
int a=10; //no
}
在C++11中,结构体或类的数据成员在申明时可以直接赋予一个默认值,初始化的方式有两种,一是使用等号“=”,二是使用大括号列表初始化的方式。注意,使用参考如下代码:
1 | class C |
注意:小括号初始化方式不能应用于就地初始化。
1.2 就地初始化与初始化列表的先后顺序
C++11标准支持了就地初始化非静态数据成员的同时,初始化列表的方式也被保留下来,也就是说既可以使用就地初始化,也可以使用初始化列表来完成数据成员的初始化工作。当二者同时使用时,并不冲突,初始化列表发生在就地初始化之后,即最终的初始化结果以初始化列表为准。参考如下代码:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
using namespace std;
class Mem
{
public:
Mem(int i,int j):m1(i),m2(j) {}
int m1 = 1;
int m2 = {2};
};
int main()
{
Mem mem(11,22);
cout<<"m1="<< mem.m1<<" m2="<<mem.m2<<endl;
}
程序输出结果:1
m1=11 m2=22
2.初始化列表
C++11之前主要有以下几种初始化方式:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22//小括号初始化
string str("hello");
//等号初始化
string str="hello";
//POD对象与POD数组列表初始化
struct Studnet
{
char* name;
int age;
};
Studnet s={"dablelv",18}; //纯数据(Plain of Data,POD)类型对象
Studnet sArr[]={{"dablelv",18},{"tommy",19}}; //POD数组
//构造函数的初始化列表
class Class
{
int x;
public:
Class():x(0){}
};
这么多的对象初始化方式,不仅增加了学习成本,也使得代码风格有较大出入,影响了代码的可读性和统一性。从C++11开始,对列表初始化(List Initialization)的功能进行了扩充,可以作用于任何类型对象的初始化,至此,列表初始化方式完成了天下大一统。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17class Test
{
int a;
int b;
public:
Test(int i, int j);
};
Test t{0,0}; //C++11 only,相当于 Test t(0,0);
Test* pT=new Test{1,2}; //C++11 only,相当于 Test* pT=new Test{1,2};
int* a = new int[3]{1,2,0}; //C++11 only
class X {
int a[4];
public:
X() : a{1,2,3,4} {} //C++11 only, 初始化数组成员
};
此外,C++11列表初始化还可以应用于容器,终于可以摆脱 push_back() 调用了,C++11中可以直观地初始化容器:1
2
3//C++11 container initializer
vector<string> vs={"first", "second", "third"};
map<string,string> singers ={{"Lady Gaga", "+1 (212) 555-7890"},{"Beyonce Knowles", "+1 (212) 555-0987"}};