delete 和 delete [] 的真正區別

2021-02-21 CPP開發者

(點擊上方公眾號,可快速關注)

來源:cbNotes,2014-08-28

連結:http://blog.csdn.net/cbnotes/article/details/38900799

如有好的文章投稿,請點擊 → 這裡查看詳情;

c++中對new申請的內存的釋放方式有delete和delete[]兩種方式,到底這兩者有什麼區別呢?

1.我們通常從教科書上看到這樣的說明:

delete 釋放new分配的單個對象指針指向的內存

delete[] 釋放new分配的對象數組指針指向的內存

那麼,按照教科書的理解,我們看下下面的代碼:

int *a = new int[10];

delete a;        //方式1

delete [] a;     //方式2

肯定會有很多人說方式1肯定存在內存洩漏,是這樣嗎?

(1). 針對簡單類型 使用new分配後的不管是數組還是非數組形式內存空間用兩種方式均可 如:

int *a = new int[10];

delete a;

delete [] a;

此種情況中的釋放效果相同,原因在於:分配簡單類型內存時,內存大小已經確定,系統可以記憶並且進行管理,在析構時,系統並不會調用析構函數,

它直接通過指針可以獲取實際分配的內存空間,哪怕是一個數組內存空間(在分配過程中 系統會記錄分配內存的大小等信息,此信息保存在結構體_CrtMemBlockHeader中,

具體情況可參看VC安裝目錄下CRTSRCDBGDEL.cpp)

(2). 針對類Class,兩種方式體現出具體差異

當你通過下列方式分配一個類對象數組:

class A

   {

   private:

      char *m_cBuffer;

      int m_nLen;

   public:

      A(){ m_cBuffer = new char[m_nLen]; }

      ~A() { delete [] m_cBuffer; }

   };

   A *a = new A[10];

   delete a;         //僅釋放了a指針指向的全部內存空間 但是只調用了a[0]對象的析構函數 剩下的從a[1]到a[9]這9個用戶自行分配的m_cBuffer對應內存空間將不能釋放 從而造成內存洩漏

   delete [] a;      //調用使用類對象的析構函數釋放用戶自己分配內存空間並且   釋放了a指針指向的全部內存空間

所以總結下就是,如果ptr代表一個用new申請的內存返回的內存空間地址,即所謂的指針,那麼:

delete   ptr   代表用來釋放內存,且只用來釋放ptr指向的內存。 

delete[]   rg   用來釋放rg指向的內存,!!還逐一調用數組中每個對象的destructor!!

對於像int/char/long/int*/struct等等簡單數據類型,由於對象沒有destructor,所以用delete 和delete [] 是一樣的!但是如果是C++對象數組就不同了!

關於 new[] 和 delete[],其中又分為兩種情況:(1) 為基本數據類型分配和回收空間;(2) 為自定義類型分配和回收空間。 

對於 (1),上面提供的程序已經證明了 delete[] 和 delete 是等同的。但是對於 (2),情況就發生了變化。

我們來看下面的例子,通過例子的學習了解C++中的delete和delete[]的使用方法

#include <iostream>

using namespace std;

/////////class Babe

class Babe

{

public:

    Babe()

    {

        cout << \"Create a Babe to talk with me\" << endl;

    }

    ~Babe()

    {

        cout << \"Babe don\'t Go away,listen to me\" << endl;

    }

};

//////////main function

int main()

{

    Babe* pbabe = new Babe[3];

    delete pbabe;

    pbabe = new Babe[3];

    delete pbabe[];

    return 0;

}

結果是:

Create a babe to talk with me

 

Create a babe to talk with me

 

Create a babe to talk with me

 

Babe don\'t go away,listen to me

 

Create a babe to talk with me

 

Create a babe to talk with me

 

Create a babe to talk with me

 

Babe don\'t go away,listen to me

 

Babe don\'t go away,listen to me

 

Babe don\'t go away,listen to me

大家都看到了,只使用delete的時候只出現一個 Babe don’t go away,listen to me,而使用delete[]的時候出現3個 Babe don’t go away,listen to me。不過不管使用delete還是delete[]那三個對象的在內存中都被刪除,既存儲位置都標記為可寫,但是使用delete的時候只調用了pbabe[0]的析構函數,而使用了delete[]則調用了3個Babe對象的析構函數。你一定會問,反正不管怎樣都是把存儲空間釋放了,有什麼區別。答:關鍵在於調用析構函數上。此程序的類沒有使用作業系統的系統資源(比如:Socket、File、Thread等),所以不會造成明顯惡果。如果你的類使用了作業系統資源,單純把類的對象從內存中刪除是不妥當的,因為沒有調用對象的析構函數會導致系統資源不被釋放,如果是Socket則會造成Socket資源不被釋放,最明顯的就是埠號不被釋放,系統最大的埠號是65535(216 _ 1,因為還有0),如果埠號被佔用了,你就不能上網了,呵呵。如果File資源不被釋放,你就永遠不能修改這個文件,甚至不能讀這個文件(除非註銷或重器系統)。如果線程不被釋放,這它總在後臺運行,浪費內存和CPU資源。這些資源的釋放必須依靠這些類的析構函數。所以,在用這些類生成對象數組的時候,用delete[]來釋放它們才是王道。而用delete來釋放也許不會出問題,也許後果很嚴重,具體要看類的代碼了.

覺得本文有幫助?請分享給更多人

關注「CPP開發者」

看更多精選C/C++技術文章

↓↓↓

相關焦點

  • malloc/free與new/delete的區別
    malloc與free是C++/C語言的標準庫函數,new/delete是C++的運算符。它們都可用於申請動態內存和釋放內存。下面來看他們的區別。一、操作對象有所不同。
  • 資料庫中 truncate與delete的區別你了解多少?
    要知道truncate和delete的區別,首先我們應該知道它們是什麼 truncate [/trket/],delete [/dlit/].一。語法:(1)truncate: TRUNCATE [TABLE] tbl_name 。
  • SQL Server中DELETE和TRUNCATE的區別
    DELETE和TRUNCATE語句之間的區別是求職面試中最常見的問題之一。這兩條語句都可以從表中刪除數據。然而,也有不同之處。TRUNCATEDELETE從表中刪除所有記錄。我們不能使用WHERE刪除特定的記錄刪除所有記錄,並可以使用WHERE刪除特定記錄。
  • Oracle資料庫知識點分享:truncate和delete區別
    一、知識點簡介Truncate和delete在實際應用中都是使用比較頻繁的,他們都是刪除表數據的語句,在這裡我們整理一下兩者之間的區別,以方便大家在應用中選擇。二、知識點內容1、 語句類型Truncate屬於SQL DDL語言。
  • delete後加 limit是個好習慣麼 !
    作者:_陳哈哈https://blog.csdn.net/qq_39390545/article/details/107519747在業務場景要求高的資料庫中,對於單條刪除和更新操作,在 delete 和 update 後面加 limit 1 絕對是個好習慣。
  • delete後加 limit是個好習慣麼 ?
    ,在 delete 和 update 後面加 limit 1 絕對是個好習慣。delete from t where sex = 1 limit 100; 你或許沒有用過,在一般場景下,我們對 delete 後是否需要加 limit 的問題很陌生,也不知有多大區別,今天帶你來了解一下,記得 mark!
  • delete後加 limit是個好習慣麼?
    本文轉載自【微信公眾號:java進階架構師,ID:java_jiagoushi】經微信公眾號授權轉載,如需轉載與原文作者聯繫在業務場景要求高的資料庫中,對於單條刪除和更新操作,在 delete 和 update 後面加 limit 1 絕對是個好習慣。
  • Linux下malloc/free與new/delete的區別
    由於malloc/free是庫函數,不是運算符,他們不能執行構造函數和析構函數,只是負責分配內存。  1、malloc與free是C++/C語言的標準庫函數,new/delete是C++的運算符。它們都可用於申請動態內存和釋放內存。
  • delete後加 limit是個好習慣麼
    ,在delete和update後面加limit 1絕對是個好習慣。delete from t where sex = 1 limit 100; 你或許沒有用過,在一般場景下,我們對 delete 後是否需要加 limit 的問題很陌生,也不知有多大區別,今天帶你來了解一下,記得mark!
  • 面試題:Mysql中drop、delete與truncate有什麼區別?
    最明顯的區別:(1) DELETE語句執行刪除的過程是每次從表中刪除一行,並且同時將該行的刪除操作作為事務記錄在日誌中保存以便進行進行回滾操作
  • delete、truncate、drop的區別有哪些,該如何選擇
    5、對於delete from table_name where xxx 帶條件的刪除, 不管是InnoDB還是MyISAM都不會釋放磁碟空間;6、delete操作以後使用 optimize table table_name 會立刻釋放磁碟空間。
  • C++ delete具體用法簡述
    C++ delete具體用法簡述 C++ delete主要是作用域動態分配以及撤銷內存的運算符。那麼它與new的用法之間有什麼聯繫呢?在這裡我們將會一一進行介紹。
  • 面試被問到SQL | delete、truncate、drop 有什麼區別?
    現在來介紹另外兩個兄弟,都是刪除表數據的,其實也是很容易理解的 老大-drop出沒場合:drop table  tb --tb表示數據表的名字,下同絕招:刪除內容和定義,釋放空間。簡單來說就是把整個表去掉.以後要新增數據是不可能的,除非新增一個表,       例如:一個班就是一個表,學生就是表中的數據,學生的職務就是定義       drop table class,就是把整個班移除.學生和職務都消失比如下面TestSchool資料庫中有兩張表[Classes]表和[Teacher]表
  • 面試被問delete後有必要加 limit麼 ?
    ,在 delete 和 update 後面加 limit 1 絕對是個好習慣。delete from t where sex = 1 limit 100; 你或許沒有用過,在一般場景下,我們對 delete 後是否需要加 limit 的問題很陌生,也不知有多大區別,今天帶你來了解一下,記得 mark!
  • 美亞的close listing和delete product and listing有...
    美亞的close listing和delete product and listing有什麼區別? 當我想在manage inventory頁面編輯一款產品時,有兩個可用的選項:close listing、delete product and listing,這兩個有什麼區別呢?
  • 不懂就問:delete 後加 limit 是個好習慣麼 ?
    https://chensj.blog.csdn.net/在業務場景要求高的資料庫中,對於單條刪除和更新操作,在 delete 和 update 後面加 limit 1 絕對是個好習慣。那麼,在日常執行 delete 時,我們是否需要養成加 limit 的習慣呢?是不是一個好習慣呢?在日常的 SQL 編寫中,你寫 delete 語句時是否用到過以下 SQL?
  • 了解各種不同意義的new和delete
    如果你希望對象產生於 heap,就調用我們平時使用的 new operator(new操作符),它不但分配內存並且為該對象調用一個構造函數;如果你只打算分配內存,請調用 operator new,這樣和調用 malloc差不多,只會分配內存,不會調用構造函數
  • MySQL中delete和update語句的用法
    昨天和大家分享學習了insert,今天我們一起學習一下刪除與更新,即delete和update的用法。where eid=2;再關掉查詢,刷新表格,看一下數據,已經更新:再來刪除一個語句,同樣,刪除語句的時候,也一定要加上where條件,不然整張表格都會受到影響,比如我們刪除部門編號為1的記錄:刪除語句為:delete
  • C+使用new與delete時,你注意到這3個地方了嗎?
    C++的動態內存管理是通過new和delete兩個操作來完成的,即用new來申請空間,用delete來釋放空間。在使用new和delete時,注意以下原則。
  • MySQL恢復delete的數據
    這世界上有後悔藥-www.houhuiyao.cc 後悔藥數據恢復 站長語前面介紹了MySQL資料庫在使用InnoDB引擎時,如果誤刪了數據表,在共享表空間MySQL數據表InnoDB引擎表誤刪恢復(共享表空間ibdata1)和獨立表空間