new
和delete
不是函數(shù),和sizeof
一樣都是C++定義的關(guān)鍵字,不同的是sizeof
在編譯時(shí)就可以確定其返回值,而new
和delete
相對(duì)復(fù)雜
示例
string *ps = new string("hello world");
如果換做c語(yǔ)言,上面這句話就會(huì)變成:
char *ps = (char *)malloc(sizeof(char)*12);
ps = "hello world";
這里就可以看出new
和mallocc
的幾點(diǎn)不同:
malloc
申請(qǐng)完空間后不會(huì)對(duì)內(nèi)存進(jìn)行必要的初始化,而new
可以new
操作符內(nèi)存分配成功時(shí),返回的是對(duì)象類型的指針,類型嚴(yán)格與對(duì)象匹配,無(wú)須進(jìn)行類型轉(zhuǎn)換,即new
是類型安全性的操作符;而malloc內(nèi)存分配成功后返回的是void*
,需要通過強(qiáng)制類型轉(zhuǎn)換,將通用類型指針void*
轉(zhuǎn)換成所需要的指針new
操作符申請(qǐng)內(nèi)存分配時(shí)無(wú)須指定內(nèi)存塊的大小,編譯器會(huì)根據(jù)類型信息自行計(jì)算;而malloc
需要顯示地寫出所需內(nèi)存塊的大小new和malloc更多的不同請(qǐng)參考此文章
總結(jié)表
特征 | new/delete | malloc/free |
---|---|---|
分配內(nèi)存的位置 | 自由存儲(chǔ)區(qū) | 堆 |
內(nèi)存分配成功的返回值 | 完整類型指針 | void* |
內(nèi)存分配失敗的返回值 | 默認(rèn)拋出異常 | 返回NULL |
分配內(nèi)存的大小 | 由編譯器根據(jù)類型計(jì)算得出 | 必須顯式指定字節(jié)數(shù) |
處理數(shù)組 | 有處理數(shù)組的new版本new[] | 需要用戶計(jì)算數(shù)組的大小后進(jìn)行內(nèi)存分配 |
已分配內(nèi)存的擴(kuò)充 | 無(wú)法直觀地處理 | 使用realloc簡(jiǎn)單完成 |
是否相互調(diào)用 | 可以,看具體的operator new/delete實(shí)現(xiàn) | 不可調(diào)用new |
分配內(nèi)存時(shí)內(nèi)存不足 | 客戶能夠指定處理函數(shù)或重新制定分配器 | 無(wú)法通過用戶代碼進(jìn)行處理 |
函數(shù)重載 | 允許 | 不允許 |
構(gòu)造函數(shù)與析構(gòu)函數(shù) | 調(diào)用 | 不調(diào)用 |
通過示例說明:
class A
{private:
int var;
FILE *file;
public:
A(int v): var(v) {fopen_s(&file, "test", "r");
}
~A(){fclose(file);
}
};
類A中有兩個(gè)私有成員,一個(gè)構(gòu)造函數(shù)和析構(gòu)函數(shù),構(gòu)造函數(shù)根據(jù)傳遞參數(shù)初始化var并且打開文件,析構(gòu)函數(shù)關(guān)閉文件
我們使用下面代碼創(chuàng)建一個(gè)類的對(duì)象,返回其指針pa
A *pa = new A(10);
如下圖所示new完成的工作:
可以將new實(shí)例化對(duì)象的過程分為三步:
那么delete會(huì)做什么呢?
delete pa;
如下圖所示:
即將delete一個(gè)對(duì)象的過程也可以分兩步:
常用的動(dòng)態(tài)分配一個(gè)數(shù)組方法
string *psa = new string[10];
int *pia = new int[10];
上面在申請(qǐng)數(shù)組的時(shí)候都用到了new []
表達(dá)式,第一個(gè)數(shù)組是string類型,在分配了保存對(duì)象的內(nèi)存空間(10個(gè)string的大?。⒄{(diào)用string類的默認(rèn)構(gòu)造函數(shù)來依次初始化每個(gè)元素,最后返回第一個(gè)string的地址作為string數(shù)組的地址;第二個(gè)數(shù)組是int類型的,int是內(nèi)置類型不存在構(gòu)造函數(shù),所以new的過程中,不存在初始化,只分配了10個(gè)int類型的內(nèi)存空間。
如果想釋放空間,則使用下面語(yǔ)句
delete [] psa;
delete [] pia;
都用到了delete []
表達(dá)式,注意這個(gè)[]
一般情況下不能漏下。釋放string數(shù)組的空間時(shí),先對(duì)數(shù)組內(nèi)的每個(gè)元素都調(diào)用析構(gòu)函數(shù)析構(gòu)對(duì)象,再釋放掉整個(gè)數(shù)組的空間;而在釋放int數(shù)組時(shí),因?yàn)椴淮嬖谖鰳?gòu)函數(shù),所以會(huì)直接釋放整個(gè)int數(shù)組的空間。
可以看到delete
的[]
中并沒有填數(shù)組的大小,那么delete關(guān)鍵字怎么知道需要調(diào)用析構(gòu)函數(shù)多少次呢?
回到new [size]
,我們new一個(gè)對(duì)象數(shù)組時(shí),還需要保存數(shù)組的維度,c++的做法是在分配數(shù)組空間時(shí)多分配4個(gè)字節(jié),專門保存數(shù)組的大小,在delete []
時(shí)就可以取出這個(gè)保存的數(shù),就知道需要調(diào)用析構(gòu)函數(shù)多少次了。
依舊以類A為例,
A *pAa = new A[3];
發(fā)生過程如下圖:
注意到,在申請(qǐng)數(shù)組對(duì)象的上面確實(shí)多分配了4個(gè)字節(jié)用來保存數(shù)組的大小,但是最終返回的地址(指針)是指向第一個(gè)數(shù)組元素的。
在釋放空間時(shí):
delete []pAa;
發(fā)生的過程如下圖:
要注意的是,先從目標(biāo)地址的前4個(gè)字節(jié)中,取出數(shù)作為調(diào)用析構(gòu)函數(shù)的次數(shù),依次析構(gòu)數(shù)組內(nèi)的元素;最后在釋放內(nèi)存空間的時(shí)候,傳遞給operator delete[]()
的參數(shù)是pAa-4
,即還要釋放前面4個(gè)字節(jié)
經(jīng)過上面的分別對(duì)new/delete和new[]/delete[]的使用,可以得知這兩對(duì)之間一般情況下不能拆開隨意組合,不然會(huì)導(dǎo)致嚴(yán)重的內(nèi)存泄露/重復(fù)釋放問題:
string *psa = new string[10];
delete psa;
如果delete沒有后面的[]
意味它只會(huì)析構(gòu)一次,那么剩下的9個(gè)string對(duì)象和上面的4字節(jié)數(shù)將永遠(yuǎn)不會(huì)被釋放,當(dāng)數(shù)組很大時(shí)會(huì)造成很嚴(yán)重的內(nèi)存泄露;相反如果是new/delete[]會(huì)導(dǎo)致重復(fù)釋放內(nèi)存的問題。
一般情況下意味也有特殊情況,如下所示:
int *pia = new int[10];
delete pia;
這個(gè)操作又是合理的,因?yàn)椴顒e在int和string不同,int是內(nèi)置類型,不存在構(gòu)造和析構(gòu)函數(shù),也就是說new[]
的時(shí)候多分配的4個(gè)字節(jié),是因?yàn)閐elete的時(shí)候需要知道調(diào)用析構(gòu)函數(shù)的次數(shù),但是當(dāng)對(duì)象類型都沒有析構(gòu)函數(shù)時(shí),也就沒有多分配這4個(gè)字節(jié)的必要。直接delete pia
給operator delete
傳遞的參數(shù)就是pia
的值(數(shù)組第一個(gè)元素的地址),直接釋放所分配的內(nèi)存塊大小即可,無(wú)需析構(gòu)。
參考文章
淺談 C++ 中的 new/delete 和 new[]/delete[]
你是否還在尋找穩(wěn)定的海外服務(wù)器提供商?創(chuàng)新互聯(lián)www.cdcxhl.cn海外機(jī)房具備T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確流量調(diào)度確保服務(wù)器高可用性,企業(yè)級(jí)服務(wù)器適合批量采購(gòu),新人活動(dòng)首月15元起,快前往官網(wǎng)查看詳情吧
分享文章:C++——new和delete關(guān)鍵字-創(chuàng)新互聯(lián)
文章出自:http://jinyejixie.com/article44/coiohe.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供App開發(fā)、網(wǎng)站維護(hù)、云服務(wù)器、營(yíng)銷型網(wǎng)站建設(shè)、關(guān)鍵詞優(yōu)化、品牌網(wǎng)站設(shè)計(jì)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容