列表初始化 分析initializer_list 的實現


 

1. 統一初始化(Uniform Initialization)

(1)在C++11之前,很多程序員特別是初學者對如何初始化一個變量或對象的問題很容易出現困惑。因為可以用小括號、大括號或賦值操作符等多種方式進行初始化

(2)基於這個原因,C++11引入了“統一初始化”的概念。這意味着我們可以使用{}這種通用的語法在任何需要初始化的地方。

【實例分析】初始化列表

 
#include <iostream>
#include <vector>
#include <map>
#include <complex>
using namespace std;

//編譯選項:g++ -std=c++11 test1.cpp -fno-elide-constructors

struct Test
{
    int x;
    int y;
} test  {123, 321}; //等價於test = {123,321}

int main()
{
    int i;   //未初始化
    int j{}; //j被初始化為0
    int* p;  //未初始化
    int* q{}; //j被初始化為nullptr
    
    int* a = new int{123}; //等價於int* x=new int(123);
    double b = double{12.12}; //等價於double(12.12)產生一個臨時對象,再拷貝初始化
    int* arr = new int[3]{1, 2, 3}; //C++11中新增的初始化堆上數組的方式
    std::map<std::string, int> mm {{"2", 1},{"2", 2}, {"3", 3}}; //相當於map<string,int> mm = {...};
    
    int values[]{1, 2, 3}; //等價於int values[]={1, 2, 3};
    vector<int> v{2, 3, 5, 7, 11, 13, 17};

    complex<double> c{4.0, 3.0}; //等價於c(4.0, 3.0);
cout << test.x << endl; cout << test.y << endl; return 0; }
 

2. 列表初始化的使用細節

(1)引入初始化列表(initializer-list)出現的一些模糊概念

 
//x,y究竟為0,0還是123,321?
struct A
{
  int x;
  int y;

  A(int,int):x(0), y(0){} //非聚合類型,使用{}初始化時會調用相應的構造函數
} a = {123, 321};  //a.x=0, a.y=0
 

(2)聚合類型的定義

  ①類型是一個普通類型的數組(如int[10]、char[]、long[2][3])

  ②類型是一個類(class、struct或union),且:

    A.無基類、無虛函數以及無用戶自定義的構造函數。

    B.無private或protected的普通數據成員(即非靜態數據成員)。

    C.不能有{}和=直接初始化的非靜態數據成員(“就地初始化”)

【實例分析】聚合類型與非聚合類型的初始化

 
#include <iostream>
using namespace std;

//x,y究竟為0,0還是123,321?
struct A
{
  int x;
  int y;

  A(int,int):x(0), y(0){} //非聚合類型,使用{}初始化時會調用相應的構造函數
} a = {123, 321};  //a.x=0, a.y=0

struct Base{};

//聚合類型的定義
struct Foo : public Base //不能有基類
{
private:
    double z;      //不能有private的普通成員
    static int k;  //ok,但必須在類外用int Foo::k = 0的方式初始化
public:
    Foo(int x, int y, double z):x(x),y(y),z(z) //不能有構造函數!
    {
        cout<< "Foo(int x, int y, double z)" << endl;
    } 
    
    Foo(const Foo& foo)  //不能有構造函數!
    {
        this->x = foo.x;
        this->y = foo.y;
        this->z = foo.z;
        
        cout<< "Foo(const Foo& foo)" << endl;
    }
    
    int x;
    int y; //不能通過int y=0;或int y{0}來"就地初始化"
    virtual void F(){}; //不能有虛函數!
        
};

int main()
{
    Foo f1(1, 2, 3.0);     //直接調用構造函數初始化
    Foo f2{4, 5, 6.0};     //由於Foo是個非聚合類型,使用{}時會調用相應的構造函數來初始化。
    Foo f3 = {7, 8, 9.0};  //非聚合類型會調用構造函數來初始化
    
    cout <<"a.x = " << a.x << ", a.y = " << a.y << endl;
    
    return 0;
}
/*輸出結果
Foo(int x, int y, double z)
Foo(int x, int y, double z)
Foo(int x, int y, double z)
a.x = 0, a.y = 0
*/
 

(3)注意事項

  ①聚合類型的定義是非遞歸的。簡單來說,當一個類的普通成員是非聚合類型時,這個類也有可能是聚合類型,也就是說可以直接用列表初始化。

  ②對於一個聚合類型,可以直接使用{}進行初始化,這時相當於對其中每個元素分別賦值;而對於非聚合類型,則需要先自定義一個合適的構造函數才能使用{}進行初始化,此時使用初始化列表將調用它對應的構造函數。

 

1. 初始化列表的實現

(1)當編譯器看到{t1,t2…tn}時便會生成一個initializer_list<T>對象(其中的T為元素的類型),它關聯到一個array<T,n>

(2)對於聚合類型,編譯器會將array<T,n>內的元素逐一分解並賦值給被初始化的對象。這相當於為該對象每個字段分別賦值

(3)對於非聚合類型。如果該類存在一個接受initializer_list<T>類型的構造函數,則初始化時會將initializer_list<T>對象作為一個整體傳給構造函數。如果不存在這樣的構造函數,則array內的元素會被編譯器分解並傳給相應的能接受這些參數的構造函數(比如列表中有2個元素的,就傳給帶2個參數的構造函數。有3個元素的,就傳給帶3個參數的構造函數,依此類推……)。

【實例分析】initializer_list<T>初體驗

 
#include <iostream>
#include <vector>
#include <map>
#include <complex>
using namespace std;

//編譯選項:g++ -std=c++11 test1.cpp -fno-elide-constructors

class Foo
{
public:
    Foo(int)
    {
        cout << "Foo(int)"<< endl;
    }
    
    Foo(int, int)
    {
        cout << "Foo(int, int)"<< endl;
    }

    Foo(const Foo& f)
    {
        cout << "Foo(const Foo& f)"<< endl;
    }
};

int main()
{
    Foo f1(123);
    Foo f2 = 123;   //先將調用Foo(int)將123轉為Foo對象,再調用拷貝構造函數(后面這步可能被優化)
    Foo f3 = {123}; //生成initializer_list<int>,然后分解元素后,由於列表中只有1個元素,所以將其傳給Foo(int)
    Foo f4 = {123, 321}; //生成initializer_list<int>,然后分解元素后,由於列表中有兩個元素,所以將其傳給Foo(int, int)
    
    //編譯器會為以下花括號形成一個initializer_list<string>,背后有個array<string,6>
    //調用vector<string>的構造函數時,編譯器會找到一個接受initializer_list<string>
    //的重載的構造函數。所有的容器均有這樣的構造函數。在這個構造函數里會利用
    //initializer_list<string>來初始化。
    vector<string> city{"Berlin", "New York", "London", "Cairo","Tokyo", "Cologne"};
    
    //編譯器會為以下花括號形成一個initializer_list<double>,背后有個array<double,2>。
    //調用complex<double>的構造函數時,array內的2個元素被分解並傳給
    //Comlex<double>(double,double)這個帶有兩個參數的構造函數。因為comlex<double>並無
    //任何接受initializer_list的構造函數。
    complex<double> c{4.0, 3.0}; //等價於c(4.0, 3.0)
    
    return 0;
}
 

2. initializer_list<T>模板

//initializer_list<T>源碼分析

 
#include <iostream>

template <class T>
class initializer_list
{
public:
    typedef T         value_type;
    typedef const T&  reference; //注意說明該對象永遠為const,不能被外部修改!
    typedef const T&  const_reference;
    typedef size_t    size_type;
    typedef const T*  iterator;  //永遠為const類型
    typedef const T*  const_iterator;
private:
    iterator    _M_array; //用於存放用{}初始化列表中的元素
    size_type   _M_len;   //元素的個數
    
    //編譯器可以調用private的構造函數!!!
    //構造函數,在調用之前,編譯會先在外部准備好一個array,同時把array的地址傳入模板
    //並保存在_M_array中
    constexpr initializer_list(const_iterator __a, size_type __l)
    :_M_array(__a),_M_len(__l){};  //注意構造函數被放到private中!

    constexpr initializer_list() : _M_array(0), _M_len(0){} // empty list,無參構造函數
    
    //size()函數,用於獲取元素的個數
    constexpr size_type size() const noexcept {return _M_len;}
    
    //獲取第一個元素
    constexpr const_iterator begin() const noexcept {return _M_array;}
    
    //最后一個元素的下一個位置
    constexpr const_iterator end() const noexcept
    {
        return begin() + _M_len;
    }  
};
 

(1)initializer_list是一個輕量級的容器類型,內部定義了iterator等容器必需的概念,本質上是一個迭代器

(2)對於std:: initializer_list<T>而言,它可以接收任意長度的初始化列表,但要求元素必須是同種類型(T或可轉換為T)。

(3)它有3個成員函數:size()、begin()和end()

(4)擁有一個無參構造函數,可以被直接實例化,此時將得到一個空的列表。之后可以進行賦值操作,如initializer_list<int> list; list={1,2,3,4,5};

(5)initializer_list<T>在進行復制或賦值時,它內部將保存着列表的地址保存在_M_array中,它進行的是淺拷貝,並不真正復制每個元素,因此效率很高。

【編程實驗】打印初始化列表的每個元素

 
#include <iostream>

//打印初始化列表的每個元素
void print(std::initializer_list<int> vals)
{
    //遍歷列表中的每個元素
    for(auto p = vals.begin(); p!=vals.end(); ++p){
        std::cout << *p << " ";
    }
    
    std::cout << std::endl;
}

//std::initializer_list<T>的淺拷貝。以下的返回值應改為std
//以下的返回值應改為std::vector<int>類型,而不是std::initializer_list<int>類型。
std::initializer_list<int> func(void)
{
    int a = 1;
    int b = 2;
    
    return {a, b}; //編譯器看到{a, b}時,會做好一個array<int,2>對象(其生命
                   //期直至func結束),然后再產生一個initializer_list<int>
                   //臨時對象,由於initializer_list<int>采用的是淺拷貝,當
                   //函數返回后array<int,2>會被釋放,所以無法獲取到列表中的元素!
}

int main()
{
    print({1,2,3,4,5,6,7,8,9,10});
    
    print(func());
    
    return 0;
}
/*測試結果:
e:\Study\C++11\7>g++ -std=c++11 test1.cpp
e:\Study\C++11\7>a.exe
1 2 3 4 5 6 7 8 9 10
*/
 

3. 讓自定義的類可以接受任意長度初始化列表

(1)自定義類中重載一個可接受initializer_list<T>類型的構造函數

(2)在該構造函數中,遍歷列表元素並賦值給相應的字段。

【編程實驗】自定義類的初始化列表

 
#include <iostream>
#include <map>

using namespace std;

class Foo
{
public:
    Foo(int a, int b)
    {
        cout << "Foo(int a, int b)" << endl;
    }
    
    Foo(initializer_list<int> list)
    {
        cout << "Foo(initializer_list<int> list) : ";
        
        for(auto i : list){
            cout <<i<< " ";
        }
        
        cout << endl;
    }

};

class FooMap
{
    std::map<int, int> content;
    using pair_t = std::map<int, int>::value_type;
public:
    FooMap(std::initializer_list<pair_t> list)
    {
        for(auto it = list.begin(); it!=list.end(); ++it){
            content.insert(*it);
            
            std::cout << "{" << (*it).first <<"," <<(*it).second <<"}" << " ";
        }
        
        std::cout << std::endl;
    }
};

int main()
{
    Foo f1(77, 5);     //Foo(int a, int b), a = 77, b = 5;
    
    //注意:由於定義了Foo(initializer_list<int> list)函數,以下3種方
    //式的初始化都會將{...}作為一個整體傳遞給該函數。如果沒有定義該函
    //數,則由於該類是個非聚合類用{}初始化時,會調用構造函數來初始化。
    //但由於Foo類不存在3個參數的構造函數,所以f3那行會編譯失敗!
    Foo f2{77, 5};     //Foo(initializer_list<int> list)
    Foo f3{77, 5, 42}; //Foo(initializer_list<int> list)
    Foo f4 = {77, 5};  //Foo(initializer_list<int> list)
    
    FooMap fm = {{1,2}, {3,4},{5,6}};
    
    return 0;
}
/*測試結果:
e:\Study\C++11\7>g++ -std=c++11 test2.cpp
e:\Study\C++11\7>a.exe
Foo(int a, int b)
Foo(initializer_list<int> list) : 77 5
Foo(initializer_list<int> list) : 77 5 42
Foo(initializer_list<int> list) : 77 5
{1,2} {3,4} {5,6}
*/

參考文章:

C++11之initialization_list


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM