• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            8.8 構造函數(2) [翻譯]

            原文 from www.learncpp.com/cpp-tutorial/88-constructors-part-ii/

            私有構造函數

            偶爾,我們不想讓用戶在class外使用特殊的構造函數。為了實現這個想法,我們可以將構造函數設定為私有的。

               1: class Book
               2: {
               3: private:
               4:     int m_nPages;
               5:  
               6:     // This constructor can only be used by Book's members
               7:     Book() // private default constructor
               8:     {
               9:          m_nPages = 0;
              10:     }
              11:  
              12: public:
              13:     // This constructor can be used by anybody
              14:     Book(int nPages) // public non-default constructor
              15:     {
              16:         m_nPages = nPages;
              17:     }
              18: };
              19:  
              20: int main()
              21: {
              22:     Book cMyBook; // fails because default constructor Book() is private
              23:     Book cMyOtherBook(242); // okay because Book(int) is public
              24:  
              25:     return 0;
              26: }

            public的構造函數具有的一個問題是它們沒有提供任何控制一個特殊的類能夠被創建多少次的方法。如果一個public構造函數存在,就能夠人用戶的愿望實例化足夠多的類對象。通常限制用戶只能夠實例化一個特殊的類是有用的。有很多方式實現單例(singletons),最常用的是使用私有或保護的構造函數。

             

            Constructor chaining and initialization issues

            當你實例化一個新的對象的時候,對象構造函數被編譯器隱式調用。有兩個狀況可能是新手經常犯錯的地方:

            1)有時候,一個類具有一個能夠做另一個構造函數相同工作的構造函數,并增加額外的工作。處理器讓一個構造函數調用另一個構造函數,叫做constructor chaining。如果你嘗試這樣做,它能夠通過編譯,但是它將不能夠正確的工作,然后你就得花費大量的時間去找出原因,甚至使用debugger。但是,構造函數允許調用非構造函數。通常要小心,使用已經初始化的變量。

            (譯者加:這里C++中為什么不能使用constructor chaining呢,可以跟蹤一下下面這段代碼答案就出來了:

               1: #include <iostream>
               2: using namespace std;
               3:  
               4: class Foo 
               5: {
               6:     int m_nValue;
               7: public:
               8:     Foo() { m_nValue = 1; }
               9:     
              10:     // 這里沒用到a,因為只是一個示例
              11:     Foo(int a) { Foo(); }
              12:     
              13:     ~Foo() { cout << "destructed" << endl; }
              14:     
              15:     int getValue() { return m_nValue; }
              16: };
              17:  
              18: int main()
              19: {
              20:     Foo cTmp(1);
              21:     cout << cTmp.getValue1();
              22: }

            盡管你能將一個構造函數中的代碼復制到另一個構造函數中,重復性的代碼會使得你的類變得臃腫。最好的解決方法是創建一個非構造函數實現公共的初始化。

               1: class Foo
               2: {
               3: public:
               4:     Foo()
               5:     {
               6:         // code to do A
               7:     }
               8:  
               9:     Foo(int nValue)
              10:     {
              11:         // code to do A
              12:         // code to do B
              13:     }
              14: };

            變為

               1: class Foo
               2: {
               3: public:
               4:     Foo()
               5:     {
               6:         DoA();
               7:     }
               8:  
               9:     Foo(int nValue)
              10:     {
              11:         DoA();
              12:         // code to do B
              13:     }
              14:  
              15:     void DoA()
              16:     {
              17:         // code to do A
              18:     }
              19: };

            2)也許你想要一個成員函數實現將成員變量設定為默認值。因為你可能已經通過構造函數實現了,你也許會在成員函數中調用構造函數。但是這在C++中是不合法的。

            類似1)中實現的方式

               1: class Foo
               2: {
               3: public:
               4:     Foo()
               5:     {
               6:         Init();
               7:     }
               8:  
               9:     Foo(int nValue)
              10:     {
              11:         Init();
              12:         // do something with nValue
              13:     }
              14:  
              15:     void Init()
              16:     {
              17:         // code to init Foo
              18:     }
              19: };

            如果Init中有動態分配內存的情況,需要更加小心的對待了。在執行相應的操作之前,應該進行一定的檢測。避免錯誤的發生。

            posted on 2012-05-30 22:06 鐘謝偉 閱讀(1212) 評論(0)  編輯 收藏 引用

            <2012年6月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            1234567

            導航

            統計

            常用鏈接

            留言簿(1)

            隨筆檔案

            IT網站

            My Friends

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            色婷婷综合久久久久中文一区二区| 亚洲精品无码专区久久久| 婷婷久久香蕉五月综合加勒比| 精品国产91久久久久久久a| 激情伊人五月天久久综合| 国内精品伊人久久久久AV影院| 亚洲国产精品无码成人片久久| 久久午夜无码鲁丝片秋霞| 久久人妻少妇嫩草AV蜜桃| 午夜精品久久久久久影视777| 久久综合九色综合久99| 亚洲精品NV久久久久久久久久| 欧美一区二区久久精品| 一本一道久久综合狠狠老| 中文字幕人妻色偷偷久久| 亚洲精品国精品久久99热一| 久久综合给久久狠狠97色| 少妇内射兰兰久久| 久久久久国产精品嫩草影院| 狠狠色婷婷久久一区二区| 国产精品久久久久免费a∨| 久久伊人色| 久久人人爽人人爽人人片av麻烦 | 国产99久久久国产精免费| 久久精品不卡| 亚洲国产精品无码久久久蜜芽| 国产精品一久久香蕉产线看| 久久精品夜色噜噜亚洲A∨| 久久久久亚洲AV综合波多野结衣 | 亚洲国产二区三区久久| 伊人久久大香线蕉综合5g| 国内精品久久久久久99蜜桃| 久久精品国产清自在天天线| 亚洲中文字幕久久精品无码APP| 久久精品国产亚洲欧美| 久久无码国产专区精品| 青青青国产精品国产精品久久久久| 久久综合精品国产一区二区三区 | 久久国产成人| 潮喷大喷水系列无码久久精品| 久久青青草原精品国产不卡|