成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

構造與析構:C++對象背后的生死較量

開發
C++編譯器就像是一位貼心的管家,當你只寫了一個析構函數時,它會默默地為你準備好所有需要的"禮物" !

在C++的奇妙世界里,構造函數和析構函數就像是一對可愛的舞臺搭檔 - 構造函數負責熱情地喊出"歡迎光臨!",而析構函數則優雅地說著"后會有期~"。它們就像是照看對象的盡職保姆 ,從出生到離別的每一刻都不離不棄,默默守護著對象的整個生命周期。這對搭檔雖然經常"斗嘴" ,但卻配合得天衣無縫,為我們的程序演繹著最動人的代碼故事。

默認構造函數的神奇魔法

你知道嗎?C++編譯器就像是一位貼心的管家 ??,當你只寫了一個析構函數時,它會默默地為你準備好所有需要的"禮物" !這些禮物包括默認構造函數、拷貝構造函數、移動構造函數(C++11的新玩具 ),以及它們的賦值運算符小伙伴們。

來看看這個有趣的派對場景:

class Party {
public:
    ~Party() { /* 收拾派對現場 */ }  // 你只負責打掃就好

    // 以下函數由編譯器自動生成
    Party();                          // 默認構造函數
    Party(const Party&);              // 拷貝構造函數
    Party(Party&&);                   // 移動構造函數
    Party& operator=(const Party&);   // 拷貝賦值運算符
    Party& operator=(Party&&);        // 移動賦值運算符
};

// 瞧瞧管家為我們準備的這些精彩玩法 ??
Party p1;                    // 開啟新派對!??
Party p2(p1);               // 復制一個一模一樣的派對 ??
Party p3 = std::move(p1);   // 把派對搬到新地方 ??
p2 = p3;                    // 把派對方案復制一份 ??
p2 = std::move(p3);         // 派對場地大轉移 ??

有趣的是,我們的管家還很節儉呢!如果你沒用到某個功能,比如從沒搬過派對場地,管家就不會為移動構造函數操心。這就是所謂的"按需服務",多貼心啊!

默認構造函數的神奇魔法

你一定會好奇,為什么C++要這么貼心地幫我們準備這些默認函數呢?這就像是準備一場完美派對 - 當你說"我要收拾派對現場"(定義析構函數)的時候,C++就會想:"哎呀,既然要收拾,那一定是開過派對的吧!" 

所以它會自動幫你準備好開派對的所有必需品(默認構造函數),復制派對方案的工具(拷貝構造函數),甚至還有搬家用的箱子(移動構造函數)。這些都是為了確保我們的對象能夠快樂地誕生 、成長、搬家,最后優雅地說再見 。

這就像是一個全套的生命服務,缺一不可 。因為在C++的世界里,有始就要有終,有終就必須有始,這是一個完整的生命周期呀!

所以,盡管你只定義了析構函數,C++依然會為你生成一個默認構造函數,確保你的Party對象能夠順利地被創建。就像一個無聲的英雄,默默地為你的代碼保駕護航。

總之,C++的構造函數和析構函數就像是派對的開場和謝幕,雖然你可能只關注了謝幕,但開場的精彩同樣不容錯過!

虛析構函數 - 繼承體系中的安全衛士

在繼承關系中,析構函數是否聲明為虛函數變得尤為重要。讓我們通過一個小例子來看看為什么需要虛析構函數:

class Animal {
public:
    ~Animal() { 
        std::cout << "再見,動物!" << std::endl; 
    }
};

class Dog : public Animal {
public:
    ~Dog() { 
        std::cout << "再見,小狗!" << std::endl; 
    }
};

int main() {
    Animal* pet = new Dog();  // 通過基類指針指向派生類對象 ??
    delete pet;               // 糟糕!只會調用 Animal 的析構函數 ??
}

在上面的例子中,delete pet 只會調用Animal 的析構函數,而不會調用Dog 的析構函數。這會導致Dog 類中可能存在的資源沒有被釋放,從而引發內存泄漏。

讓我們來修復這個問題:

class Animal {
public:
    virtual ~Animal() {  // 添加 virtual 關鍵字 ?
        std::cout << "再見,動物!" << std::endl; 
    }
};

class Dog : public Animal {
public:
    ~Dog() override {    // 使用 override 更清晰 ??
        std::cout << "再見,小狗!" << std::endl; 
    }
};

int main() {
    Animal* pet = new Dog();
    delete pet;  // 現在會正確調用 Dog 的析構函數,然后是 Animal 的析構函數 ??
}

通過將Animal 的析構函數聲明為虛函數,delete pet 會首先調用Dog 的析構函數,然后調用Animal 的析構函數,確保所有資源都被正確釋放。這樣就不會有內存泄漏的問題啦!

為什么需要虛析構函數? 

在繼承關系中,使用基類指針指向派生類對象時,如果基類的析構函數不是虛函數,刪除該指針時只會調用基類的析構函數,而不會調用派生類的析構函數。這會導致派生類中分配的資源沒有被正確釋放,從而引發內存泄漏。??

析構順序的秘密

你可能會問:"為什么聲明為虛函數后,會依次調用 Dog 和 Animal 的析構函數呢?不是已經重寫了嗎?" 讓我們來揭開這個秘密:

class Animal {
protected:
    int* animalResource;  // 基類的資源 ???
public:
    Animal() { animalResource = new int(1); }
    virtual ~Animal() { 
        delete animalResource;
        std::cout << "再見,動物!" << std::endl; 
    }
};

class Dog : public Animal {
private:
    int* dogResource;    // 派生類的資源 ??
public:
    Dog() { dogResource = new int(2); }
    ~Dog() override { 
        delete dogResource;
        std::cout << "再見,小狗!" << std::endl; 
    }
};

這是因為在 C++ 中,派生類對象的析構過程遵循特定的順序:

  • 首先調用派生類(Dog)的析構函數
  • 然后自動調用基類(Animal)的析構函數

這個過程是自動且必然的,原因如下:

(1) 內存布局:Dog 對象不僅包含自己的成員(dogResource),還包含從 Animal 繼承來的所有成員(animalResource)

(2) 資源清理:

  • Dog 的析構函數負責清理 Dog 特有的資源
  • Animal 的析構函數負責清理繼承來的資源
  • 如果不調用基類的析構函數,基類的資源就會泄露

(3) 執行順序:就像蓋房子和拆房子

  • 蓋房子時是從下往上(先構造基類,再構造派生類)
  • 拆房子時是從上往下(先析構派生類,再析構基類)

所以當我們執行:

Animal* pet = new Dog();
delete pet;

輸出會是:

再見,小狗!    // 先清理 Dog 的資源
再見,動物!    // 再清理 Animal 的資源

這不是普通的函數重寫,而是 C++ 特有的析構機制,確保對象的完整清理。就像拆房子必須從頂層開始拆一樣,析構也必須從派生類開始,層層向下進行!

普通函數重寫 vs 析構函數

讓我們來對比一下普通虛函數的重寫和析構函數的區別:

class Animal {
public:
    // 普通虛函數
    virtual void speak() {
        std::cout << "動物在說話" << std::endl;
    }
    
    // 析構函數
    virtual ~Animal() {
        std::cout << "再見,動物!" << std::endl;
    }
};

class Dog : public Animal {
public:
    // 普通函數重寫 - 只會調用這個版本
    void speak() override {
        std::cout << "汪汪汪!" << std::endl;
    }
    
    // 析構函數 - 會調用這個,然后自動調用基類版本
    ~Dog() override {
        std::cout << "再見,小狗!" << std::endl;
    }
};

int main() {
    Animal* pet = new Dog();
    
    pet->speak();    // 輸出:汪汪汪!
    delete pet;      // 輸出:再見,小狗! 再見,動物!
}
  • 普通函數重寫:完全替換基類的版本,只會執行派生類的實現
  • 析構函數:是一個特殊的過程,會依次執行派生類和基類的析構函數

這種區別的設計是有意義的:

  • 普通函數重寫:我們希望完全替換掉基類的行為
  • 析構函數:我們需要清理整個繼承鏈上的所有資源,不能遺漏

性能考慮

添加虛析構函數會帶來一些開銷:

  • 每個對象都會多一個虛函數表指針(vptr) 
  • 類的大小會增加(通常是一個指針的大小) 
  • 虛函數調用比普通函數調用稍慢 

但是相比于內存泄漏的風險,這點開銷是值得的!

最佳實踐 

  • 如果你的類將被繼承,請將析構函數聲明為虛函數 
  • 如果你的類不會被繼承,則不需要虛析構函數
  • 在聲明虛析構函數時,建議使用override 關鍵字(C++11及以后) 

通過遵循這些最佳實踐,你的代碼將更加健壯,避免不必要的內存泄漏問題。

責任編輯:趙寧寧 來源: everystep
相關推薦

2011-06-15 09:47:14

C++

2009-08-14 17:24:28

C#構造函數和析構函數

2025-02-18 00:08:00

代碼C++RAII

2011-07-15 01:29:39

C++析構函數

2009-09-03 13:14:55

C#構造函數C#析構函數

2010-02-04 16:39:26

C++析構函數

2010-01-18 15:53:27

C++析構函數

2009-07-30 15:24:13

C#析構函數C#構造函數

2021-01-17 16:29:51

C++Python語言

2010-02-05 13:35:19

C++虛析構函數

2024-12-19 14:42:15

C++內存泄漏內存管理

2010-07-20 09:52:27

Perl構造函數

2010-01-27 10:13:22

C++類對象

2015-05-25 10:52:49

2009-09-02 10:49:46

C#調用析構方法

2024-04-28 11:01:27

C++編程語言函數

2009-08-13 17:30:30

C#構造函數

2012-08-15 13:31:02

筆試題

2011-07-20 13:40:09

拷貝構造函數

2010-01-27 17:16:52

C++構造函數
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美中文字幕 | 九九热国产视频 | 成年人在线观看视频 | 亚洲高清av在线 | 国产一区二区三区在线 | 精品国产一区二区三区性色av | 99久久国产 | 国产精品欧美一区二区 | 亚洲另类自拍 | 成人精品鲁一区一区二区 | 亚洲精选一区二区 | 色爱区综合| 在线电影日韩 | 国产露脸国语对白在线 | 亚洲成人精品 | 欧美片网站免费 | 欧美视频中文字幕 | 久久精品国产99国产精品 | 亚洲一区久久久 | 一区二区免费 | 99国产视频| 在线免费亚洲视频 | 亚洲成人蜜桃 | 欧美日韩不卡合集视频 | 日韩在线小视频 | 二区中文字幕 | 中文在线播放 | 国产一区二区三区在线 | 久久精品中文字幕 | 黄色一级视频 | 国产欧美日韩在线观看 | 91中文在线观看 | 99国内精品| 欧美乱码精品一区二区三区 | 91在线视频免费观看 | 亚洲日本一区二区 | h小视频| 男人天堂网址 | 日本中文字幕一区 | 国产一卡二卡三卡 | 国产精品久久久久久婷婷天堂 |