C++告訴咱們在回收用 new 分配的單個對象的內存空間的時候用 delete,回收用 new[] 分配的一組對象的內存空間的時候用 delete[]。 關於 new[] 和 delete[],其中又分爲兩種狀況:(1) 爲基本數據類型分配和回收空間;(2) 爲自定義類型分配和回收空間。
請看下面的程序。
複製代碼 代碼以下:
#include <iostream>;
using namespace std;
class T {
public:
T() { cout << "constructor" << endl; }
~T() { cout << "destructor" << endl; }
};
int main()
{
const int NUM = 3;
T* p1 = new T[NUM];
cout << hex << p1 << endl;
// delete[] p1;
delete p1;
T* p2 = new T[NUM];
cout << p2 << endl;
delete[] p2;
}
你們能夠本身運行這個程序,看一看 delete p1 和 delete[] p1 的不一樣結果,我就不在這裏貼運行結果了。
從運行結果中咱們能夠看出,delete p1 在回收空間的過程當中,只有 p1[0] 這個對象調用了析構函數,其它對象如 p1[1]、p1[2] 等都沒有調用自身的析構函數,這就是問題的癥結所在。若是用 delete[],則在回收空間以前全部對象都會首先調用本身的析構函數。 基本類型的對象沒有析構函數,因此回收基本類型組成的數組空間用 delete 和 delete[] 都是應該能夠的;可是對於類對象數組,只能用 delete[]。對於 new 的單個對象,只能用 delete 不能用 delete[] 回收空間。 因此一個簡單的使用原則就是:new 和 delete、new[] 和 delete[] 對應使用。
個人理解,當用delete來釋放用new int[]申請的內存空間時,因爲其爲基本數據類型沒有析構函數,因此使用delete與delete []相同,二者都會釋放申請的內存空間,如果自定義的數據類型,有析構函數時,用new []申請的空間,必需要用delete []來釋放,由於要delete []時會逐一調用對象數組的析構函數,而後釋放空間,若是用delete,則只會調用第一個對象的析構函數,後面對象的析構函數沒有被調用,那麼其空間是否釋放了呢??
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安裝目錄下CRT\SRC\DBGDEL.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指針指向的所有內存空間 而且調用使用類對象的析構函數釋放用戶本身分配內存空間
在 vc 中對於簡單數據類型int,char等的數組使用delete仍是delete[]是徹底同樣的,不過對於類的實例的數組來講就必須使用delete[],不然可能會引發該類的析溝函數不會被正確調用。
以上內容引自:http://www.cnblogs.com/zhengyuhong/articles/delete1.html
總結:在不明確指針指向的是不是數組仍是單個變量時,用delete[] p;來刪除指針是最好的辦法。不會引發上面說的間接內存泄露。