1.就地初始化
1.1 簡介
在C++11之前,只能對結構體或類的靜態常量成員就行就地初始化,其他的不行。如下代碼所示:
class Test
{
private:
static const int a=10; //yes
int a=10; //no
}
在C++11中,結構體或類的數據成員在申明時可以直接賦予一個默認值,初始化的方式有兩種,一是使用等號“=”,二是使用大括號列表初始化的方式。注意,使用參考如下代碼:
class C
{
private:
int a=7; //C++11 only
int b{7}; //或int b={7}; C++11 only
int c(7); //error
};
注意:小括號初始化方式不能應用於就地初始化。
1.2 就地初始化與初始化列表的先後順序
C++11標準支持了就地初始化非靜態數據成員的同時,初始化列表的方式也被保留下來,也就是說既可以使用就地初始化,也可以使用初始化列表來完成數據成員的初始化工作。當二者同時使用時,並不衝突,初始化列表發生在就地初始化之後,即最終的初始化結果以初始化列表爲準。參考如下代碼:
#include <iostream>
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;
}
程序輸出結果:
m1=11 m2=22
2.初始化列表
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開始,對列表初始化(List Initialization)的功能進行了擴充,可以作用於任何類型對象的初始化,至此,列表初始化方式完成了天下大一統。
class 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
此外,C++11列表初始化還可以應用於容器,終於可以擺脫 push_back() 調用了,C++11中可以直觀地初始化容器:
//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"}};