天天看點

C++類構造函數初始化清單 建議直接使用初始化清單為成員進行初始化

構造函數初始化清單以一個冒号開始,接着是以逗号分隔的資料成員清單,每個資料成員後面跟一個放在括号中的初始化式。例如:

class CExample {

public:

    int a;

    float b;

    //構造函數初始化清單

    CExample(): a(0),b(8.8)

    {}

    //構造函數内部指派

    CExample()

    {

        a=0;

        b=8.8;

    }

};

上面的例子中兩個構造函數的結果是一樣的。上面的構造函數(使用初始化清單的構造函數)顯式的初始化類的成員;而沒使用初始化清單的構造函數是對類的成員指派,并沒有進行顯式的初始化。

初始化和指派對内置類型的成員沒有什麼大的差別,像上面的任一個構造函數都可以。對非内置類型成員變量,為了避免兩次構造,推薦使用類構造函數初始化清單。但有的時候必須用帶有初始化清單的構造函數:

1.成員類型是沒有預設構造函數的類。若沒有提供顯示初始化式,則編譯器隐式使用成員類型的預設構造函數,若類沒有預設構造函數,則編譯器嘗試使用預設構造函數将會失敗。

2.const成員或引用類型的成員。因為const對象或引用類型隻能初始化,不能對他們指派。 

初始化資料成員與對資料成員指派的含義是什麼?有什麼差別?

首先把資料成員按類型分類并分情況說明:

1.内置資料類型,複合類型(指針,引用)

    在成員初始化清單和構造函數體内進行,在性能和結果上都是一樣的

2.使用者定義類型(類類型)

    結果上相同,但是性能上存在很大的差别。因為類類型的資料成員對象在進入函數體前已經構造完成,也就是說在成員初始化清單處進行構造對象的工作,調用構造函數,在進入函數體之後,進行的是對已經構造好的類對象的指派,又調用個拷貝指派操作符才能完成(如果并未提供,則使用編譯器提供的預設按成員指派行為)

Note:

初始化清單的成員初始化順序:

    C++初始化類成員時,是按照聲明的順序初始化的,而不是按照出現在初始化清單中的順序。

    Example:

class CMyClass {

    CMyClass(int x, int y);

    int m_x;

    int m_y;

};

CMyClass::CMyClass(int x, int y) : m_y(y), m_x(m_y)

{

}

你可能以為上面的代碼将會首先做m_y=I,然後做m_x=m_y,最後它們有相同的值。但是編譯器先初始化m_x,然後是m_y,,因為它們是按這樣的順序聲明的。結果是m_x将有一個不可預測的值。有兩種方法避免它,一個是總是按照你希望它們被初始化的順序聲明成員,第二個是,如果你決定使用初始化清單,總是按照它們聲明的順序羅列這些成員。這将有助于消除混淆。