青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

  C++博客 :: 首頁 :: 聯系 ::  :: 管理
  163 Posts :: 4 Stories :: 350 Comments :: 0 Trackbacks

常用鏈接

留言簿(48)

我參與的團隊

搜索

  •  

積分與排名

  • 積分 - 401640
  • 排名 - 59

最新評論

閱讀排行榜

評論排行榜

第9章 類的構造函數、析構函數與賦值函數

    構造函數、析構函數與賦值函數是每個類最基本的函數。它們太普通以致讓人容易麻痹大意,其實這些貌似簡單的函數就象沒有頂蓋的下水道那樣危險。
    每個類只有一個析構函數和一個賦值函數,但可以有多個構造函數(包含一個拷貝構造函數,其它的稱為普通構造函數)。對于任意一個類A,如果不想編寫上述函數,C++編譯器將自動為A產生四個缺省的函數,如
    A(void);                    // 缺省的無參數構造函數
    A(const A &a);                // 缺省的拷貝構造函數
    ~A(void);                    // 缺省的析構函數
    A & operate =(const A &a);    // 缺省的賦值函數

這不禁讓人疑惑,既然能自動生成函數,為什么還要程序員編寫?
原因如下:
(1)如果使用“缺省的無參數構造函數”和“缺省的析構函數”,等于放棄了自主“初始化”和“清除”的機會,C++發明人Stroustrup的好心好意白費了。
(2)“缺省的拷貝構造函數”和“缺省的賦值函數”均采用“位拷貝”而非“值拷貝”的方式來實現,倘若類中含有指針變量,這兩個函數注定將出錯。
   
對于那些沒有吃夠苦頭的C++程序員,如果他說編寫構造函數、析構函數與賦值函數很容易,可以不用動腦筋,表明他的認識還比較膚淺,水平有待于提高。
本章以類String的設計與實現為例,深入闡述被很多教科書忽視了的道理。String的結構如下:
    class String
    {
      public:
        String(const char *str = NULL);    // 普通構造函數
        String(const String &other);    // 拷貝構造函數
        ~ String(void);                    // 析構函數
        String & operate =(const String &other);    // 賦值函數
      private:
        char      *m_data;                // 用于保存字符串
    };
9.1 構造函數與析構函數的起源
    作為比C更先進的語言,C++提供了更好的機制來增強程序的安全性。C++編譯器具有嚴格的類型安全檢查功能,它幾乎能找出程序中所有的語法問題,這的確幫了程序員的大忙。但是程序通過了編譯檢查并不表示錯誤已經不存在了,在“錯誤”的大家庭里,“語法錯誤”的地位只能算是小弟弟。級別高的錯誤通常隱藏得很深,就象狡猾的罪犯,想逮住他可不容易。
    根據經驗,不少難以察覺的程序錯誤是由于變量沒有被正確初始化或清除造成的,而初始化和清除工作很容易被人遺忘。Stroustrup在設計C++語言時充分考慮了這個問題并很好地予以解決:把對象的初始化工作放在構造函數中,把清除工作放在析構函數中。當對象被創建時,構造函數被自動執行。當對象消亡時,析構函數被自動執行。這下就不用擔心忘了對象的初始化和清除工作。
    構造函數與析構函數的名字不能隨便起,必須讓編譯器認得出才可以被自動執行。Stroustrup的命名方法既簡單又合理:讓構造函數、析構函數與類同名,由于析構函數的目的與構造函數的相反,就加前綴‘~’以示區別。
除了名字外,構造函數與析構函數的另一個特別之處是沒有返回值類型,這與返回值類型為void的函數不同。構造函數與析構函數的使命非常明確,就象出生與死亡,光溜溜地來光溜溜地去。如果它們有返回值類型,那么編譯器將不知所措。為了防止節外生枝,干脆規定沒有返回值類型。(以上典故參考了文獻[Eekel, p55-p56])
9.2 構造函數的初始化表
    構造函數有個特殊的初始化方式叫“初始化表達式表”(簡稱初始化表)。初始化表位于函數參數表之后,卻在函數體 {} 之前。這說明該表里的初始化工作發生在函數體內的任何代碼被執行之前。
    構造函數初始化表的使用規則:
?    如果類存在繼承關系,派生類必須在其初始化表里調用基類的構造函數。
例如
    class A
    {…
        A(int x);        // A的構造函數
};   
    class B : public A
    {…
        B(int x, int y);// B的構造函數
    };
    B::B(int x, int y)
     : A(x)                // 在初始化表里調用A的構造函數
    {
      …
}   
?    類的const常量只能在初始化表里被初始化,因為它不能在函數體內用賦值的方式來初始化(參見5.4節)。
?    類的數據成員的初始化可以采用初始化表或函數體內賦值兩種方式,這兩種方式的效率不完全相同。
    非內部數據類型的成員對象應當采用第一種方式初始化,以獲取更高的效率。例如
    class A
{…
    A(void);                // 無參數構造函數
    A(const A &other);        // 拷貝構造函數
    A & operate =( const A &other);    // 賦值函數
};

    class B
    {
      public:
        B(const A &a);    // B的構造函數
      private:   
        A  m_a;            // 成員對象
};

示例9-2(a)中,類B的構造函數在其初始化表里調用了類A的拷貝構造函數,從而將成員對象m_a初始化。
示例9-2 (b)中,類B的構造函數在函數體內用賦值的方式將成員對象m_a初始化。我們看到的只是一條賦值語句,但實際上B的構造函數干了兩件事:先暗地里創建m_a對象(調用了A的無參數構造函數),再調用類A的賦值函數,將參數a賦給m_a。

B::B(const A &a)
 : m_a(a)           
{
   …
}    B::B(const A &a)
{
m_a = a;

}
 示例9-2(a) 成員對象在初始化表中被初始化      示例9-2(b) 成員對象在函數體內被初始化

對于內部數據類型的數據成員而言,兩種初始化方式的效率幾乎沒有區別,但后者的程序版式似乎更清晰些。若類F的聲明如下:
class F
{
  public:
    F(int x, int y);        // 構造函數
  private:
    int m_x, m_y;
    int m_i, m_j;
}
示例9-2(c)中F的構造函數采用了第一種初始化方式,示例9-2(d)中F的構造函數采用了第二種初始化方式。

F::F(int x, int y)
 : m_x(x), m_y(y)           
{
   m_i = 0;
   m_j = 0;
}    F::F(int x, int y)
{
   m_x = x;
   m_y = y;
   m_i = 0;
   m_j = 0;
}
 示例9-2(c) 數據成員在初始化表中被初始化     示例9-2(d) 數據成員在函數體內被初始化
9.3 構造和析構的次序
    構造從類層次的最根處開始,在每一層中,首先調用基類的構造函數,然后調用成員對象的構造函數。析構則嚴格按照與構造相反的次序執行,該次序是唯一的,否則編譯器將無法自動執行析構過程。
一個有趣的現象是,成員對象初始化的次序完全不受它們在初始化表中次序的影響,只由成員對象在類中聲明的次序決定。這是因為類的聲明是唯一的,而類的構造函數可以有多個,因此會有多個不同次序的初始化表。如果成員對象按照初始化表的次序進行構造,這將導致析構函數無法得到唯一的逆序。[Eckel, p260-261]
9.4 示例:類String的構造函數與析構函數
    // String的普通構造函數
    String::String(const char *str)
{
    if(str==NULL)
    {
        m_data = new char[1];
        *m_data = ‘\0’;
    }   
    else
    {
        int length = strlen(str);
        m_data = new char[length+1];
        strcpy(m_data, str);
    }
}   

// String的析構函數
    String::~String(void)
{
    delete [] m_data;   
// 由于m_data是內部數據類型,也可以寫成 delete m_data;
    }
9.5 不要輕視拷貝構造函數與賦值函數
    由于并非所有的對象都會使用拷貝構造函數和賦值函數,程序員可能對這兩個函數有些輕視。請先記住以下的警告,在閱讀正文時就會多心:
?    本章開頭講過,如果不主動編寫拷貝構造函數和賦值函數,編譯器將以“位拷貝”的方式自動生成缺省的函數。倘若類中含有指針變量,那么這兩個缺省的函數就隱含了錯誤。以類String的兩個對象a,b為例,假設a.m_data的內容為“hello”,b.m_data的內容為“world”。
現將a賦給b,缺省賦值函數的“位拷貝”意味著執行b.m_data = a.m_data。這將造成三個錯誤:一是b.m_data原有的內存沒被釋放,造成內存泄露;二是b.m_data和a.m_data指向同一塊內存,a或b任何一方變動都會影響另一方;三是在對象被析構時,m_data被釋放了兩次。

?    拷貝構造函數和賦值函數非常容易混淆,常導致錯寫、錯用。拷貝構造函數是在對象被創建時調用的,而賦值函數只能被已經存在了的對象調用。以下程序中,第三個語句和第四個語句很相似,你分得清楚哪個調用了拷貝構造函數,哪個調用了賦值函數嗎?
String  a(“hello”);
String  b(“world”);
String  c = a;    // 調用了拷貝構造函數,最好寫成 c(a);
c = b;     // 調用了賦值函數
本例中第三個語句的風格較差,宜改寫成String c(a) 以區別于第四個語句。
9.6 示例:類String的拷貝構造函數與賦值函數
    // 拷貝構造函數
    String::String(const String &other)
    {   
// 允許操作other的私有成員m_data
    int length = strlen(other.m_data);   
    m_data = new char[length+1];
    strcpy(m_data, other.m_data);
}

// 賦值函數
    String & String::operate =(const String &other)
    {   
        // (1) 檢查自賦值
        if(this == &other)
            return *this;
       
        // (2) 釋放原有的內存資源
        delete [] m_data;
       
        // (3)分配新的內存資源,并復制內容
    int length = strlen(other.m_data);   
    m_data = new char[length+1];
        strcpy(m_data, other.m_data);
       
        // (4)返回本對象的引用
        return *this;
}   
   
    類String拷貝構造函數與普通構造函數(參見9.4節)的區別是:在函數入口處無需與NULL進行比較,這是因為“引用”不可能是NULL,而“指針”可以為NULL。
    類String的賦值函數比構造函數復雜得多,分四步實現:
(1)第一步,檢查自賦值。你可能會認為多此一舉,難道有人會愚蠢到寫出 a = a 這樣的自賦值語句!的確不會。但是間接的自賦值仍有可能出現,例如
   
// 內容自賦值
b = a;

c = b;

a = c;        // 地址自賦值
b = &a;

a = *b;

也許有人會說:“即使出現自賦值,我也可以不理睬,大不了化點時間讓對象復制自己而已,反正不會出錯!”
他真的說錯了。看看第二步的delete,自殺后還能復制自己嗎?所以,如果發現自賦值,應該馬上終止函數。注意不要將檢查自賦值的if語句
if(this == &other)
錯寫成為
    if( *this == other)
(2)第二步,用delete釋放原有的內存資源。如果現在不釋放,以后就沒機會了,將造成內存泄露。
(3)第三步,分配新的內存資源,并復制字符串。注意函數strlen返回的是有效字符串長度,不包含結束符‘\0’。函數strcpy則連‘\0’一起復制。
(4)第四步,返回本對象的引用,目的是為了實現象 a = b = c 這樣的鏈式表達。注意不要將 return *this 錯寫成 return this 。那么能否寫成return other 呢?效果不是一樣嗎?
不可以!因為我們不知道參數other的生命期。有可能other是個臨時對象,在賦值結束后它馬上消失,那么return other返回的將是垃圾。
9.7 偷懶的辦法處理拷貝構造函數與賦值函數
    如果我們實在不想編寫拷貝構造函數和賦值函數,又不允許別人使用編譯器生成的缺省函數,怎么辦?
    偷懶的辦法是:只需將拷貝構造函數和賦值函數聲明為私有函數,不用編寫代碼。
例如:
    class A
    { …
      private:
        A(const A &a);                // 私有的拷貝構造函數
        A & operate =(const A &a);    // 私有的賦值函數
    };

如果有人試圖編寫如下程序:
    A  b(a);    // 調用了私有的拷貝構造函數
    b = a;        // 調用了私有的賦值函數
編譯器將指出錯誤,因為外界不可以操作A的私有函數。
9.8 如何在派生類中實現類的基本函數
    基類的構造函數、析構函數、賦值函數都不能被派生類繼承。如果類之間存在繼承關系,在編寫上述基本函數時應注意以下事項:
?    派生類的構造函數應在其初始化表里調用基類的構造函數。
?    基類與派生類的析構函數應該為虛(即加virtual關鍵字)。例如
#include <iostream.h>
class Base
{
  public:
    virtual ~Base() { cout<< "~Base" << endl ; }
};

class Derived : public Base
{
  public:
    virtual ~Derived() { cout<< "~Derived" << endl ; }
};

void main(void)
{
    Base * pB = new Derived;  // upcast
    delete pB;
}

輸出結果為:
    ~Derived
    ~Base
如果析構函數不為虛,那么輸出結果為
    ~Base

?    在編寫派生類的賦值函數時,注意不要忘記對基類的數據成員重新賦值。例如:
class Base
{
  public:

    Base & operate =(const Base &other);    // 類Base的賦值函數
  private:
    int  m_i, m_j, m_k;
};

class Derived : public Base
{
  public:

    Derived & operate =(const Derived &other);    // 類Derived的賦值函數
  private:
    int  m_x, m_y, m_z;
};

Derived & Derived::operate =(const Derived &other)
{
    //(1)檢查自賦值
    if(this == &other)
        return *this;

    //(2)對基類的數據成員重新賦值
    Base::operate =(other);    // 因為不能直接操作私有數據成員

    //(3)對派生類的數據成員賦值
    m_x = other.m_x;
    m_y = other.m_y;
    m_z = other.m_z;

    //(4)返回本對象的引用
    return *this;
}

9.9 一些心得體會
有些C++程序設計書籍稱構造函數、析構函數和賦值函數是類的“Big-Three”,它們的確是任何類最重要的函數,不容輕視。
也許你認為本章的內容已經夠多了,學會了就能平安無事,我不能作這個保證。如果你希望吃透“Big-Three”,請好好閱讀參考文獻[Cline] [Meyers] [Murry]。




posted on 2007-12-17 16:42 sdfasdf 閱讀(338) 評論(0)  編輯 收藏 引用 所屬分類: C++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲国产精品女人久久久| 国产精品美女久久久久av超清 | 欧美激情黄色片| 久久一日本道色综合久久| 亚洲黄色三级| 亚洲视频在线二区| 国一区二区在线观看| 亚洲第一偷拍| 欧美性色aⅴ视频一区日韩精品| 亚洲一区高清| 久久国产婷婷国产香蕉| 亚洲精品国偷自产在线99热| 国产亚洲欧美另类中文| 亚洲一区二区三区在线观看视频 | 亚洲欧美成人在线| 一区二区三区在线不卡| 亚洲人午夜精品| 国产一区二区三区免费不卡| 欧美大片免费观看在线观看网站推荐| 欧美激情精品久久久久久黑人| 亚洲自拍偷拍麻豆| 免费成人高清在线视频| 亚洲综合丁香| 欧美成人免费va影院高清| 亚洲欧美欧美一区二区三区| 麻豆精品网站| 欧美在线视频在线播放完整版免费观看 | 欧美激情日韩| 国产伦精品一区二区三区视频孕妇| 欧美成年人在线观看| 国产精品网站一区| 亚洲人成网在线播放| 国产一区二区精品丝袜| 中国女人久久久| 亚洲激情视频在线播放| 午夜一区二区三区在线观看| 一区二区三区免费网站| 久久久精品2019中文字幕神马| 亚洲欧美99| 欧美区一区二| 欧美电影在线播放| 一区在线观看视频| 欧美影院精品一区| 午夜视频久久久| 欧美色欧美亚洲另类七区| 欧美xxx成人| 激情成人亚洲| 久久高清免费观看| 久久久久亚洲综合| 国产裸体写真av一区二区| 在线一区二区三区做爰视频网站| 99在线热播精品免费99热| 久久久久久久久蜜桃| 久久精品视频免费| 国产亚洲一区二区三区在线播放 | 亚洲国产影院| 亚洲国产一区二区三区青草影视| 久久精品免视看| 久久这里只有| 国语精品一区| 久久久国产亚洲精品| 久久久久综合一区二区三区| 国产一区二区三区直播精品电影| 亚洲免费在线电影| 久久久久久夜| 伊人色综合久久天天五月婷| 久久婷婷久久| 亚洲精品1区2区| 一区二区三区视频在线| 国产精品wwwwww| 亚洲欧美日韩久久精品| 欧美日韩综合一区| 午夜精品国产精品大乳美女| 国产精品美女主播在线观看纯欲| 亚洲图片在线观看| 久久久精品性| 亚洲国产高清一区| 欧美激情一区二区久久久| 日韩一本二本av| 欧美一区网站| 亚洲高清一区二区三区| 欧美日本国产视频| 亚洲欧美综合v| 欧美国产日本| 亚洲午夜一二三区视频| 国产亚洲精品久久久久婷婷瑜伽| 久久久久久亚洲综合影院红桃| 亚洲电影专区| 午夜欧美大片免费观看| 尤物九九久久国产精品的特点| 欧美大胆a视频| 亚洲一区二区av电影| 免费看亚洲片| 午夜精品美女久久久久av福利| 国外成人在线视频| 欧美日韩伦理在线免费| 欧美一区三区三区高中清蜜桃| 亚洲观看高清完整版在线观看| 亚洲丝袜av一区| 在线成人激情视频| 国产精品久久久久久久午夜片| 欧美在线一级视频| 日韩视频永久免费观看| 麻豆成人91精品二区三区| 亚洲午夜精品一区二区三区他趣| 好吊色欧美一区二区三区四区| 欧美激情精品久久久久久黑人| 亚洲欧美中文日韩v在线观看| 亚洲国产裸拍裸体视频在线观看乱了中文 | 欧美电影在线播放| 欧美一区二区三区在线观看视频| 亚洲第一综合天堂另类专| 欧美在线91| 中国亚洲黄色| 日韩视频精品在线| 尤物yw午夜国产精品视频| 国产精品一级二级三级| 欧美日韩成人一区二区| 巨胸喷奶水www久久久免费动漫| 亚洲影院高清在线| 99亚洲精品| 91久久国产综合久久| 欧美aaaaaaaa牛牛影院| 久久久激情视频| 欧美一区二区三区视频免费| 亚洲一区二区免费在线| 99人久久精品视频最新地址| 亚洲国产午夜| 亚洲国产婷婷香蕉久久久久久| 国产综合香蕉五月婷在线| 国产精品一区二区a| 国产精品乱码人人做人人爱| 欧美日韩一区在线观看| 欧美精品一区二区蜜臀亚洲| 欧美aⅴ99久久黑人专区| 久久国产精品久久久久久久久久| 性欧美video另类hd性玩具| 亚洲一区二区黄色| 午夜欧美精品久久久久久久| 亚洲视频欧美在线| 亚洲中无吗在线| 亚洲欧美中文字幕| 久久9热精品视频| 久久在线免费观看视频| 久久久综合网站| 午夜视频在线观看一区二区三区| 亚洲影音一区| 性色av一区二区三区在线观看 | 黄色成人91| 亚洲第一精品电影| 91久久精品美女高潮| 日韩亚洲精品视频| 中日韩视频在线观看| 午夜精品福利在线观看| 久久精品论坛| 欧美黑人在线播放| 99成人精品| 午夜精品一区二区三区在线播放| 欧美一级理论性理论a| 久久影音先锋| 欧美日韩美女在线| 国产欧美亚洲一区| 亚洲国产日韩在线一区模特| 日韩一级精品视频在线观看| 亚洲欧美日韩精品一区二区| 久久精品国产久精国产爱| 欧美777四色影视在线| 亚洲精品免费电影| 欧美一级成年大片在线观看| 美女精品在线观看| 欧美视频免费看| 激情文学综合丁香| 在线午夜精品自拍| 久久日韩粉嫩一区二区三区| 亚洲欧洲视频在线| 午夜精品久久99蜜桃的功能介绍| 麻豆91精品| 国产精品久久久久久妇女6080 | 国产视频亚洲| 亚洲精品一区二区三区樱花| 欧美在线视频免费观看| 亚洲国产日韩欧美在线图片| 亚洲欧美一区二区在线观看| 欧美成人精品一区| 国产区精品在线观看| 亚洲免费观看高清在线观看 | 欧美精品观看| 国产有码在线一区二区视频| 在线视频日本亚洲性| 欧美成人一区二区| 亚洲欧美国产制服动漫| 欧美激情第三页| 狠狠综合久久av一区二区小说 | 亚洲国产一区在线观看| 欧美在线亚洲一区| 日韩视频免费| 欧美成人视屏| 亚洲风情亚aⅴ在线发布| 久久精品国产免费观看| 一区二区三区成人精品|