一起來(lái)學(xué)習(xí)C++的動(dòng)態(tài)內(nèi)存管理
1.new和delete
C語(yǔ)言?xún)?nèi)存管理方式在C++中可以繼續(xù)使用,但有些地方就無(wú)能為力而且使用起來(lái)比較麻煩,因此C++又提出了自己的內(nèi)存管理方式:通過(guò)new和delete操作符進(jìn)行動(dòng)態(tài)內(nèi)存管理
這里在動(dòng)態(tài)申請(qǐng)和釋放時(shí)一定要匹配使用
但是我們發(fā)現(xiàn)即使我們隨意使用程序也沒(méi)有崩潰,但是這些開(kāi)辟的空間都是默認(rèn)類(lèi)型的
我們接下來(lái)試一下開(kāi)辟自定義類(lèi)型的空間程序是否會(huì)崩潰
我們運(yùn)行程序,發(fā)現(xiàn)直接就崩潰了
那為什么自定義的類(lèi)型就不可以隨意使用釋放內(nèi)存的函數(shù)呢?這就要探究malloc,free和new,delete的區(qū)別。我們先來(lái)看malloc和free
然后再看new和delete
2.new和delete在底層是怎么實(shí)現(xiàn)的:
2.1new底層的實(shí)現(xiàn): 我們先來(lái)new一個(gè)test類(lèi)型的空間。
然后運(yùn)行起來(lái)我們轉(zhuǎn)到反匯編代碼看看
總結(jié)new的實(shí)現(xiàn)就是調(diào)用operate new(size_t)函數(shù),函數(shù)內(nèi)部循壞調(diào)用malloc如果申請(qǐng)空間成功就返回,如果申請(qǐng)失敗就檢測(cè)是否有應(yīng)對(duì)措施,如果提供就執(zhí)行措施,再繼續(xù)malloc,如果未提供就拋出異常。然后再調(diào)用構(gòu)造函數(shù)將申請(qǐng)的內(nèi)存初始化。
2.2delete底層的實(shí)現(xiàn): 我們執(zhí)行delete語(yǔ)句,轉(zhuǎn)到反匯編來(lái)
進(jìn)入不知名函數(shù)
那么我們來(lái)看看operator delete(void *pt)函數(shù)是如何如何實(shí)現(xiàn)的
總結(jié):delete 的實(shí)現(xiàn)就是先調(diào)用類(lèi)中的析構(gòu)函數(shù),將對(duì)象中的數(shù)據(jù)清空,然后調(diào)用operator delete(void *pt)函數(shù),將內(nèi)存釋放掉。
2.3new []底層的實(shí)現(xiàn):
執(zhí)行完這一系列操作后就會(huì)調(diào)用n次構(gòu)造函數(shù),將對(duì)象構(gòu)造成功總結(jié):new []就是調(diào)用operate new[]函數(shù),在operate new []中調(diào)用operate new函數(shù)執(zhí)行剛剛的一系列操作,然后返回申請(qǐng)的內(nèi)存。再調(diào)用n次構(gòu)造函數(shù)將申請(qǐng)的內(nèi)存初始化。
2.4delete []的實(shí)現(xiàn):
總結(jié):delete函數(shù)就是先調(diào)用n次析構(gòu)函數(shù)將申請(qǐng)的內(nèi)存空間中的數(shù)據(jù)清空,然后再調(diào)用operate delete()函數(shù)將內(nèi)存釋放掉。
3.重載new和delete
既然有了new和delete這么好的申請(qǐng)內(nèi)存的方法,那我們?yōu)槭裁催€要重新實(shí)現(xiàn)new和delete呢?注意這里的重新實(shí)現(xiàn)實(shí)現(xiàn)new和delete并不是重新實(shí)現(xiàn)new和delete的申請(qǐng)內(nèi)存的方式,而是有時(shí)我們?cè)賒ebug版本下調(diào)試時(shí)需要一些打印輸出一些信息,這里我們需要再用new申請(qǐng)內(nèi)存的時(shí)候要實(shí)現(xiàn)輸出一些信息。比如說(shuō)打印文件名,調(diào)用函數(shù),調(diào)用行數(shù)等等。我們重新實(shí)現(xiàn)new將我們的文件信息,調(diào)用函數(shù),調(diào)用行數(shù)都打印出來(lái)
可以看到雖然將函數(shù)都打印出來(lái)了但是傳參的時(shí)候需要傳許多參數(shù),所以我們可以用宏替換的方式來(lái)解決
但是我們需要這些信息都是在調(diào)試的時(shí)候才需要,所以我們可以條件編譯一下只在debug版本底下執(zhí)行
我們將delete實(shí)現(xiàn)重載,
注意這里如果我們將delete自己實(shí)現(xiàn)重載之后再調(diào)用delete重載函數(shù)是不會(huì)調(diào)用類(lèi)中的析構(gòu)函數(shù)的,如果我們的對(duì)象中涉及了資源的申請(qǐng)那么就不會(huì)釋放。
4.定位new:
使用場(chǎng)景:在有些場(chǎng)景下我們可能會(huì)申請(qǐng)一塊內(nèi)存空間,但是這塊內(nèi)存空間并沒(méi)有初始化,當(dāng)我們想要使用這塊內(nèi)存空間時(shí),想給他初始化,對(duì)于類(lèi)類(lèi)型的對(duì)象我們想要給他初始化時(shí),只能調(diào)用構(gòu)造函數(shù)初始化,但是構(gòu)造函數(shù)只能在創(chuàng)建時(shí)由編譯器自動(dòng)調(diào)用(就像人不可以選擇自己的出生時(shí)間),那我們這時(shí)就要使用定位new來(lái)給已經(jīng)申請(qǐng)?zhí)柕膬?nèi)存中創(chuàng)建一個(gè)對(duì)象。
定位new的幾種使用方式:
釋放空間時(shí)
定位new的原理:
5.內(nèi)存檢測(cè)函數(shù):_CrtDumpMemoryLeaks();
_CrtDumpMemoryLeaks();函數(shù)是window操作系統(tǒng)提供的一個(gè)api(應(yīng)用程序接口)函數(shù),當(dāng)程序中有內(nèi)存泄漏時(shí)就會(huì)在底行輸出內(nèi)存泄漏信息。
我們將申請(qǐng)的空間正確釋放,可以看到?jīng)]有任何的提示。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C語(yǔ)言中的數(shù)組和指針匯編代碼分析實(shí)例
這篇文章主要介紹了C語(yǔ)言中的數(shù)組和指針匯編代碼分析實(shí)例,本文用一則C語(yǔ)言例子來(lái)得到對(duì)應(yīng)的匯編代碼,并一一注解每句匯編代碼的含義,需要的朋友可以參考下2015-06-06關(guān)于C++繼承你可能會(huì)忽視的點(diǎn)
繼承是面向?qū)ο笕筇匦灾?有些類(lèi)與類(lèi)之間存在特殊的關(guān)系,下面這篇文章主要給大家介紹了關(guān)于C++繼承你可能會(huì)忽視的點(diǎn),文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-02-02C 語(yǔ)言基礎(chǔ)之C語(yǔ)言的常見(jiàn)關(guān)鍵字
C語(yǔ)言中有一些預(yù)先定義的字符串,他們本身被賦予了自身的功能。并且我們?cè)诙x變量的時(shí)候,不能去搶他們的名字來(lái)用。他們就是今天的主角:關(guān)鍵字,下面文章將給大家做詳細(xì)介紹2021-09-09C/C++中的sizeof運(yùn)算符和size_t類(lèi)型的詳解
今天小編就為大家分享一篇關(guān)于C/C++中的sizeof運(yùn)算符和size_t類(lèi)型的詳解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-10-10C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)順序表的進(jìn)階講解
程序中經(jīng)常需要將一組數(shù)據(jù)元素作為整體管理和使用,需要?jiǎng)?chuàng)建這種元素組,用變量記錄它們,傳進(jìn)傳出函數(shù)等。一組數(shù)據(jù)中包含的元素個(gè)數(shù)可能發(fā)生變化,順序表則是將元素順序地存放在一塊連續(xù)的存儲(chǔ)區(qū)里,元素間的順序關(guān)系由它們的存儲(chǔ)順序自然表示2022-04-04OpenCV獲取鼠標(biāo)左鍵點(diǎn)擊位置圖像的像素值
這篇文章主要為大家詳細(xì)介紹了OpenCV獲取鼠標(biāo)左鍵點(diǎn)擊位置圖像的像素值,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01