C++類成員初始化順序問題

今天剛把買了一個月的劍指offer這本書翻閱一下,看到一個簡單的舉例。說應聘C++崗位的不知道成員變量初始化順序!大冬天的背後竟冒出了一絲冷汗,因爲我也不知道,所以就上網查了一下,將學到的知識記錄如下。
這裏寫圖片描述
主要參考博客:

  1. C++類成員初始化順序問題1
  2. C++類成員初始化順序問題2

問題來源:
由於面試題中,考官出了一道簡單的程序輸出結果值的題:如下,

#include <iostream>
using namespace std;

class A
{
public:
    A()
        :n2(0),
        n1(n2 + 2)
    {}

    //~A();

    void print()
    {
        cout << "n1:" << n1 << " ,  n2:" << n2 << endl;
    }

private:
    int n1;
    int n2;
};

//A::~A()
//{
//}

int main()
{
    A a;
    a.print();
    system("pause");
    return 0;
}

大家預測一下結果,在沒運行的時候,我想大多數不懂初始化成員列表的順序的人都會和我一樣認爲輸出的是:n1:2 , n2:0

(這裏有一個小插曲,之前我的VS代碼貼進去竟然總是報錯,仔細審查了幾次,都沒覺得有問題,最後重新建個工程再運行又是好的)

最後,我們可以看到輸出結果如下:
這裏寫圖片描述

如果我把A類中構造函數改成:

A()  
    {  
        n2 = 0;  
        n1 = n2 + 2;  
    }  

運行結果如下:
這裏寫圖片描述

分析原因如下:
1、成員變量在使用初始化列表初始化時,與構造函數中初始化成員列表的順序無關,只與定義成員變量的順序有關。因爲成員變量的初始化次序是根據變量在內存中次序有關,而內存中的排列順序早在編譯期就根據變量的定義次序決定了。這點在EffectiveC++中有詳細介紹。
2、如果不使用初始化列表初始化,在構造函數內初始化時,此時與成員變量在構造函數中的位置有關。
3、注意:類成員在定義時,是不能初始化的
4、注意:類中const成員常量必須在構造函數初始化列表中初始化。
5、注意:類中static成員變量,必須在類外初始化。
6、靜態變量進行初始化順序是基類的靜態變量先初始化,然後是它的派生類。直到所有的靜態變量都被初始化。這裏需要注意全局變量和靜態變量的初始化是不分次序的。這也不難理解,其實靜態變量和全局變量都被放在公共內存區。可以把靜態變量理解爲帶有“作用域”的全局變量。在一切初始化工作結束後,main函數會被調用,如果某個類的構造函數被執行,那麼首先基類的成員變量會被初始化。

總結:
變量的初始化順序就應該是:

1 基類的靜態變量或全局變量

2 派生類的靜態變量或全局變量

3 基類的成員變量

4 派生類的成員變量

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章