c 11之前主要有以下几种初始化方式:
//小括号初始化
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以前,程序员,或者初学者经常会感到疑惑关于怎样去初始化一个变量或者是一个对象。这么多的对象初始化方式,不仅增加了学习成本,也使得代码风格有较大出入,影响了代码的可读性和统一性。
从c 11开始,对列表初始化(list initialization)的功能进行了扩充,可以作用于任何类型对象的初始化,至此,列表初始化方式完成了天下大一统。
花括号列表初始化,作为c 11新标准的一部被加入到了c 中。
因为这个原因,c 11提出了统一初始化,以为着使用这初始化列表,下面的做法都是正确的。
class test
{
int a;
int b;
public:
c(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
此外,c 11列表初始化还可以应用于容器,终于可以摆脱 push_back() 调用了,c 11中可以直观地初始化容器:
//c 11 container initializer
vector vs={"first", "second", "third"};
map singers ={
{"lady gaga", " 1 (212) 555-7890"},{"beyonce knowles", " 1 (212) 555-0987"}};
因此,可以将c 11提供的列表初始化作为统一的初始化方式,既降低了记忆难度,也提高的代码的统一度。
此外,c 11中,类的数据成员在申明时可以直接赋予一个默认值:
class c
{
private:
int a=7; //c 11 only
};
所谓花括号列表初始化,即是用花括号来初始化变量,其形式如: int test = { 0 } ;无论是初始化对象还是为对象赋值 , 在c 11下都可以使用这种形式的初始值。
不同的一点 是:使用这种形式来初始化内置类型的变量时,若存在类型转换且具有丢失信息的风险时,编译器将会报错。
通过这一点可以看出,列表初始化比原有的初始化方式具有更严格的安全要求。下面是例子:
long double ld = 3.1415926536;
int a {ld} , b = {ld} // 编译器报错,存在丢失信息的风险
int c (ld) , d = ld ; //正确