還是先說說區別吧:
1. new/delete是C++裏纔有的,而new/delete與malloc/free一個顯著的區別在於,new是建造一個對象,並調用對象的構造函數來初始化對象,其實在所有的new操作過程中,總是分爲兩步的:第一步是申請內存,第二步則是調用構造函數初始化對象。同樣,在調用delete的時候,需要先調用析構函數,然後在銷燬堆內存。
2. new/delete通常來說是操作符,就是"+","-"一樣。
3. new/delete是可以重載的,而重載之後,就成爲了函數。
4. malloc在申請內存的時候,必須要提供申請的長度,而返回的指針是void*型,必須要強轉才能成爲需要的類型。
5. 當new/delete在類中被重載的時候,可以自定義申請過程,比如記錄所申請內存的總長度,以及跟蹤每個對象的指針。
6. C++默認的new/delete操作符內部,其實也調用了malloc/free這兩個函數。
共同點:
1. 都必須配對使用,這裏的配對使用,可不能理解爲一個new/malloc就對應一個delete/free,而是指在作用域內,new/malloc所申請的內存,必須被有效釋放,否則將會導致內存泄露,至於內存泄露的檢查方法,我們推薦的工具是大家衆所周知的BoundsChecker,至於如何使用BoundsChecker,我們將在以後撰文詳解。
2. 都是申請內存,釋放內存,free和delete可以釋放NULL指針。
注意點:
1. new/delete與malloc/free不能混合使用,有些人對這個觀點持懷疑態度,因爲在很多時候,他混合使用之後也沒有嚴重的後遺症,那是因爲在通常情況下,new操作符的確調用了malloc這個函數,所以free函數可以正常的釋放new出來的內存空間。但這並不能保證所有的new操作符都是調用C++的new的原始操作符,而最常見的是,在類中,我們是可以重載new這個操作符的,這樣的話,如果一但在operator=new()函數中調用了其它的申請函數的話東西,free將無法正常工作,或者說也將導致內存泄露。
舉幾個簡單的例子吧:
class CTest
{
public:
CTest();
~CTest();
private:
int* __m_pn;
};
CTest::CTest()
{
__m_pn = new int[128]; assert(__m_pn);
}
CTest::~CTest()
{
assert(__m_pn);
delete[] __m_pn;
__m_pn = NULL;
}
int main()
{
int* pn = (int*)malloc(sizeof(int));
*pn = 15;
free(pn);
pn = NULL; // 置空
free(pn); // OK,沒有問題
double* pd = new double;
*pd = 212.211;
delete = pd;
short* ps = new short[128]; // new出一個數組來
ps[1] = 1231;
ps[11] = 1111;
delete[] ps; // 請注意delete的語法。
CTest* pTest = new CTest; // new出一個對象,並初始化
// …… 幹活
delete pTest; // 析構,釋放內存
}
// 以下是一個重載new操作符的例子,一般來說是不會用到的,除非要設計一個編譯器之類的東西。
class Sample
{
public:
static CSample* operator= new()
{
CSample* p = (CSample*)malloc(size(Sample));
__m_nCount++; // 記錄這個類被申請的對象的數目
return p;
}
private:
static int __m_nCount_;
};