只有在堆內(nèi)存里面才會發(fā)生內(nèi)存泄漏的問題,在棧內(nèi)存中不會發(fā)生內(nèi)存泄漏。因?yàn)闂?nèi)存在自動(dòng)分配空間之后,還會自動(dòng)釋放空間。
什么是堆內(nèi)存?存儲方式是什么樣的呢?
首先我們先來介紹一下堆內(nèi)存在C代碼中的存儲方式。C代碼中動(dòng)態(tài)申請堆內(nèi)存的申請函數(shù)是malloc,常見的內(nèi)存代碼如下圖所示:
因?yàn)閙alloc函數(shù)返回值是一個(gè)內(nèi)存地址,所以保存堆內(nèi)存的變量一定得是一個(gè)指針,當(dāng)然這個(gè)變量可以是一個(gè)單指針,也可以是一個(gè)多重指針。
如何獲取堆內(nèi)存?
對于堆內(nèi)存的獲取方法,我們可以有兩種方法,第一種是用返回值傳遞內(nèi)存指針,第二種方法是通過參數(shù)傳遞給內(nèi)存指針。上面我們用到的malloc申請內(nèi)存,就是屬于方法一的一種具體表現(xiàn)形式,是直接把返回值傳遞給內(nèi)存指針。
方法一:把函數(shù)返回值直接賦值給指針,一般表現(xiàn)形式如下:
方法二:將指針地址作為函數(shù)返回參數(shù),通過返回參數(shù)保存堆內(nèi)存地址,一般表現(xiàn)形式如下:
總結(jié):這兩類方法的本質(zhì)是一樣的,都是函數(shù)內(nèi)存間接申請了內(nèi)存,但是只有傳遞內(nèi)存的方法不一樣,方法一是通過返回值傳遞內(nèi)存指針,方法二是通過參數(shù)傳遞內(nèi)存指針。
內(nèi)存泄漏的三個(gè)原因
當(dāng)我們的代碼出現(xiàn)內(nèi)存泄漏的時(shí)候,一般都會包含以下幾個(gè)原因:
- 函數(shù)內(nèi)有局部指針變量定義
- 對該局部指針有獲取內(nèi)存的操作
- 在函數(shù)返回前沒有釋放該內(nèi)存,也未保存到其他全局變量或返回上一級函數(shù)
如何檢查內(nèi)存泄漏
為了避免檢查內(nèi)存泄漏,我們還是要養(yǎng)成良好的編碼習(xí)慣。當(dāng)我們要進(jìn)行檢查內(nèi)存泄漏問題的時(shí)候,一般要做到以下三點(diǎn):
- 當(dāng)我們在函數(shù)中看到有局部指針的時(shí)候,一定要仔細(xì)檢查是否有存泄漏的問題發(fā)生,養(yǎng)成仔細(xì)檢查的習(xí)慣
- 如果有局部變量,并且有對局部變量賦值的操作,要檢查函數(shù)的返回的指針到底是指向什么?是全局變量、靜態(tài)數(shù)據(jù)還是堆內(nèi)存?如果代碼中有不熟悉的接口,要找到對應(yīng)接口文檔或源代碼分析,保證不要出現(xiàn)不必要的錯(cuò)誤
- 如果函數(shù)中有對局部指針有內(nèi)存申請的操作,那么要檢查被保存的是全局變量嗎?會被作為函數(shù)返回值嗎?如果都不是的話,那要排查函數(shù)所有的“return”的地方,要保證內(nèi)存被正確釋放,不占用內(nèi)存
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。
舉報(bào)投訴
-
存儲
+關(guān)注
關(guān)注
13文章
4296瀏覽量
85798 -
內(nèi)存
+關(guān)注
關(guān)注
8文章
3019瀏覽量
74003 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4327瀏覽量
62569
發(fā)布評論請先 登錄
相關(guān)推薦
堆棧內(nèi)存和堆內(nèi)存之間的區(qū)別
編寫有效的代碼需要了解堆棧和堆內(nèi)存,這使其成為學(xué)習(xí)編程的重要組成部分。不僅如此,新程序員或職場老手都應(yīng)該完全熟悉堆棧內(nèi)存和堆內(nèi)存之間的區(qū)別,
發(fā)表于 08-07 12:23
?721次閱讀
內(nèi)存管理實(shí)例中運(yùn)行Test函數(shù)會有什么樣的結(jié)果
); strcpy(str, "helloworld"); printf(str);}運(yùn)行Test函數(shù)會有什么樣的結(jié)果?答: 程序崩潰;原因:1、實(shí)參是通過拷貝的方式 傳遞給行參
發(fā)表于 01-22 16:15
【原創(chuàng)】堆內(nèi)存的那些事
區(qū)域那樣有明顯的分界線。堆內(nèi)存的釋放看下面這個(gè)圖:看到這樣頻繁的使用區(qū)域和釋放,那么很容易看出堆內(nèi)存是不連續(xù)的,跟堆
發(fā)表于 07-12 09:48
堆和棧的區(qū)別是什么
在回答完進(jìn)程的虛擬地址空間布局之后(上一篇),面試官可能抓住堆和棧深入展開。堆和棧的區(qū)別①管理方式:棧由編譯器自動(dòng)管理;堆由程序員控制,使用方便,但易產(chǎn)生
發(fā)表于 12-22 07:26
單片機(jī)下的堆和棧是什么樣的分布呢?
是基于os層來聊的。那么,在赤裸裸的單片機(jī)下的堆和棧是什么樣的分布呢?以下是網(wǎng)摘:剛接手STM32時(shí),你只編寫一個(gè)int main(){while(1);}BUILD://Prog...
發(fā)表于 01-25 07:07
ESP8266上的內(nèi)存類型有多少?
arduino 草圖時(shí),我使用什么樣的內(nèi)存?如果我存儲一些非易變的東西但程序在哪里運(yùn)行,我可以使用 SPIFFS?在公羊?我有多少內(nèi)存?
發(fā)表于 02-24 06:34
什么樣的冰箱沒有霜
什么樣的冰箱沒有霜
什么樣的冰箱比較容易結(jié)霜,什么樣的冰箱在這方面就會好些?使用中要注意哪些問題呢?
風(fēng)冷冰箱不易結(jié)霜
發(fā)表于 02-21 17:56
?2022次閱讀
未來存儲技術(shù)的發(fā)展是什么樣的
未來的存儲技術(shù)會是什么樣子呢?對于基于NVMe的傳統(tǒng)Flash技術(shù),我們應(yīng)該繼續(xù)期望更高的容量。
發(fā)表于 09-18 14:39
?1328次閱讀
5G給內(nèi)存和存儲帶來了什么樣的發(fā)展美光科技的解答
內(nèi)存和存儲的區(qū)別越來越模糊,在2021年,將看到企業(yè)正在尋求新型解決方案,例如存儲級內(nèi)存和內(nèi)存虛擬化,以進(jìn)一步釋放AI及激增的數(shù)據(jù)量帶來的
簡述C語言中的內(nèi)存泄漏的原理及解決方法
什么是堆內(nèi)存?堆內(nèi)存是如何分配的?
在一般的編譯系統(tǒng)中,堆內(nèi)存的分配方向和棧內(nèi)存是相反的。當(dāng)棧內(nèi)存從高地址向低地址增長的時(shí)候,堆內(nèi)存
程序內(nèi)存分區(qū)中的堆與棧
與棧表示兩種內(nèi)存管理方式; (2)數(shù)據(jù)結(jié)構(gòu)場景下,堆與棧表示兩種常用的數(shù)據(jù)結(jié)構(gòu)。 1.程序內(nèi)存分區(qū)中的堆與棧 1.1 棧簡介 棧由操作系統(tǒng)自
malloc 申請內(nèi)存的兩種方式
我們知道m(xù)alloc() 并不是系統(tǒng)調(diào)用,也不是運(yùn)算符,而是 C 庫里的函數(shù),用于動(dòng)態(tài)分配內(nèi)存。 malloc 申請內(nèi)存的時(shí)候,會有兩種方式向操作系統(tǒng)申請堆
jvm配置堆內(nèi)存初始值參數(shù)
JVM(Java Virtual Machine)是Java語言的運(yùn)行環(huán)境,它通過解釋字節(jié)碼并執(zhí)行相應(yīng)的指令來運(yùn)行Java程序。在JVM中,堆(Heap)是用于存儲對象實(shí)例的內(nèi)存區(qū)域。而在Java
堆和棧的區(qū)別和使用注意事項(xiàng)
堆和棧是在計(jì)算機(jī)科學(xué)中廣泛使用的兩種數(shù)據(jù)結(jié)構(gòu),它們具有不同的用途和特點(diǎn)。堆和棧的區(qū)別涉及到內(nèi)存分配、訪問方式、數(shù)據(jù)存儲等方面。在使用
評論