C++11中列表初始化機制的概念與實例詳解

概述

定義:列表初始化是C++11引入的新標準,目的是統一初始化方式

C++11以前隻能使用列表初始化來初始化內置類型數組和POD類型對象,C++11中列表初始化可以用於初始化任何類型對象

  • POD(plain old data)類型:僅由內置類型變量構成且不含指針的類,簡單來說是可以直接使用memcpy復制的對象
  • 聚合體(aggregate):聚合體一定是POD類型
    • 無自定義構造函數
    • 無私有或保護的非靜態數據成員(靜態成員與單獨對象無關,故不影響初始化)
    • 無基類
    • 無虛函數
    • 無類內已經初始化的非靜態數據成員

註意:區分列表初始化和初始化列表

        列表初始化:用{}進行初始化的方式

        初始化列表:構造函數體前對對象成員直接進行初始化的列表

        initializer_list:一種用於未定參數的輕量STL容器

實現機制詳解

對內置類型對象、POD對象和類對象的列表初始化實現細節是不同的

POD類型的列表初始化

  • 此處POD類型包括:內置類型、聚合體類
  • 內置類型數組按照順序初始化
    • C++11標準中列表初始化會防止可能導致潛在信息丟失的類型縮小(即不能像賦值一樣將大類型如int隱式轉換成小類型如char)
  • 聚合體類按照成員定義順序依次初始化

含有構造函數的類的列表初始化(C++11)

  • 通過{}進行初始化和()結果一致【即通過()調用構造函數的地方都可以完全等價地用{}代替】,都是直接用括號內的值調用對應構造函數直接初始化對象,並不會先生成臨時對象再拷貝
  • ={}與{}是等價的語法【即加不加=對初始化行為沒有影響】,均不會調用拷貝運算符或拷貝構造函數
  • 與內置類型的列表初始化一致,C++11的列表初始化隻能用於初始化,不能用於已初始化對象的賦值
  • 實際機制猜想:傳遞的實際參數為initializer_list類型,通過匹配重載函數實現調用【我不知道怎麼驗證這個過程,求大佬解答】

列表初始化用於函數返回值

  • 在返回值類型為對象(不能是對象的引用)的函數中可以返回{}的列表初始化
  • {}返回值的實際類型為initiallizer list(但不能聲明為std::initializer_list),相當於返回構造函數的表達式,因此類型不能是對象的引用

引入std::initializer_list

  • initializer_list為一個輕量級STL模板,聲明在頭文件<initializer_list>中,定義在命名空間std中
  • 任意的STL容器都與未指定長度的數組有一樣的初始化能力,可以填入任何數量的同類型數據,因此可以用STL容器輕易對固定類型的類進行賦值
  • initializer_list是一個輕量級的模板,可以接受任意長度的同類型的數據也就是接受可變長參數,同時作為STL容器它具有STL容器的共同特征(如迭代器)
    • 隻有三個成員接口:begin() end() size()
    • 隻能被整體的初始化和賦值,迭代器遍歷的數據僅可讀,不能對單個數據進行修改
  • 所有{}對象都是隱式創建的std::initializer_list類型字面量(右值),廣泛用於實現列表初始化(不需要頭文件)

代碼驗證

class testClass
{
private:
	int a;
	int b;
public:
	testClass() :a(0), b(0) {
		cout << "default init\n";
	}
	testClass(int a) :a(a), b(a) {
		cout << "sing-val init\n";
	}
	testClass(int a, int b) :a(a), b(b) {
		cout << "val init\n";
	}
	testClass(testClass& temp) :a(temp.a), b(temp.b) {
		cout << "copy init\n";
	}
	testClass& operator=(testClass& temp) {
		//testClass& newobj = *this;
		a = temp.a;
		b = temp.b;
		cout << "copy assign\n";
		return *this;
	}
	testClass& operator=(int x) {
		a = x;
		b = x;
		cout << "int-convert assign\n";
		//testClass& newobj = *this;
		return *this;
	}
	testClass& operator++() {
		a++;
		b++;
	}
	void printVal(ostream& os) {
		os << "a=" << a << "\n";
		os << "b=" << b << "\n";
	}
};
using tc = testClass;
tc& makeObj(int x, int y)
{
	return { x,y };
}
int main()
{
	tc a(1, 1); //val init
	tc b{ 1,1 }; //val init
	tc c = { 1,1 }; //val init
	tc d = tc{ 1,1 }; //val init
	cout << endl;
	tc* e = new tc[2]; //default init *2
	cout << endl;
	tc* f = new tc[3]{ {1,1},{2,2},{3,3} }; //val init *3
	cout << endl;
	tc* g = new tc[5]{ {1,1},{1} }; // val init + sing-val init + default init *3
	cout << endl;
	cout << "testing return val of init_list\n";
	tc h = makeObj(2, 2); //val init
	tc i = h; //copy init
	i = d; //copy assign
	i.printVal(cout);
	return 0;
}

以下為運行截圖

列表初始化測試

添加initializer_list為參數的構造函數後

testClass::testClass(initializer_list<int> list) :a(0), b(0)
{
	int ab = 1;
	for (auto it = list.begin(); it != list.end(); it++)
	{
		if (ab)
			a += *it;
		else
			b += *it;
	}
	cout << "init_list init\n";
}
 
int main()
{
	tc a(1, 1); //val init
	tc b{ 1,1 }; //val init
	tc c = { 1,1 }; //val init
	tc d = tc{ 1,1 }; //val init
	cout << endl;
	tc* e = new tc[2]; //default init *2
	cout << endl;
	tc* f = new tc[3]{ {1,1},{2,2},{3,3} }; //val init *3
	cout << endl;
	tc* g = new tc[5]{ {1,1},{1} }; // val init + sing-val init + default init *3
	cout << endl;
	cout << "testing return val of init_list\n";
	tc h = makeObj(2, 2); //val init
	tc i = h; //copy init
	i = d; //copy assign
	i.printVal(cout);
	cout << endl;
	cout << "testing argument init_list\n";
	tc j = { 1,2,3,4,5,6 };
	tc k = { 9 };
	return 0;
}

以下為運行截圖


添加init_list後測試截圖

由此可見所有列表初始化都調用瞭含有initializer_list為參數的構造函數,證實瞭列表初始化是基於隱式轉換並以initializer_list為底層實現的構想

應用

  • 在聲明時直接初始化堆上分配的對象(數組)
    • 類:可以顯式指定使用的構造函數(默認會執行無參數的構造函數)
    • 內置類型:可以在分配時直接指定值
  • 在函數返回對象時避免自動存儲期對象銷毀的問題
  • 手動調用std::initializer_list實現可變參數初始化

列表初始化防止類型收窄

C++11的列表初始化還有一個額外的功能就是可以防止類型收窄,也就是C++98/03中的隱式類型轉換,將范圍大的轉換為范圍小的表示,在C++98/03中類型收窄並不會編譯出錯,而在C++11中,使用列表初始化的類型收窄編譯將會報錯:

int a = 1.1; //OK
int b{ 1.1 }; //error
 
float f1 = 1e40; //OK
float f2{ 1e40 }; //error
 
const int x = 1024, y = 1;
char c = x; //OK
char d{ x };//error
char e = y;//error
char f{ y };//error

總結

列表初始化通過C++11引入的initializer_list容器實現瞭初始化方式的統一,可以看作一種語法糖

初始化類對象時,通過()調用構造函數的地方都可以完全等價地用{}代替

={}不會生成臨時對象再拷貝初始化

到此這篇關於C++11中列表初始化機制的文章就介紹到這瞭,更多相關C++11列表初始化機制內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: