内存泄露一个经典例子
這個程序測試后會有什么結(jié)果?[美國某著名計算機(jī)嵌入式公司2005年9月面試題]
解析:毛病出在函數(shù)GetMemory中。void GetMemory(char *p, int num)中的*p實際上是主
函數(shù)中str的一個副本,編譯器總是要為函數(shù)的每個參數(shù)制作臨時副本。在本例中,p申請了
新的內(nèi)存,只是把p所指的內(nèi)存地址改變了,但是str絲毫未變。因為函數(shù)GetMemory沒有返
回值,因此str并不指向p所申請的那段內(nèi)存,所以函數(shù)GetMemory并不能輸出任何東西,如
下圖所示。事實上,每執(zhí)行一次GetMemory就會申請一塊內(nèi)存,但申請的內(nèi)存卻不能有效釋
放,結(jié)果是內(nèi)存一直被獨占,最終造成內(nèi)存泄露。
如果一定要用指針參數(shù)去申請內(nèi)存,那么應(yīng)該采用指向指針的指針,傳str的地址給函數(shù)
GetMemory。代碼如下:
? ? ?這樣的話,程序就可以運行成功。字符串是一個比較特殊的例子。我們分別打印*str、
str、&str可以發(fā)現(xiàn),結(jié)果分別是h、hello、0*22f7c。str就是字符串的值;*str是字符串中某一
字符的值,默認(rèn)的是首字符,所以是h;&str是字符串的地址值。
? ? ?由于“指向指針的指針”這個概念不容易理解,我們可以用函數(shù)返回值來傳遞動態(tài)內(nèi)存。
這種方法更加簡單,代碼如下:
我們可以對這道題推而廣之,看一下整型變量是如何傳值的,代碼如下:
GetMemory2把v的地址傳了進(jìn)來,*z是地址里的值,是v的副本。通過直接修改地址里的
值,不需要有返回值,也把v給修改了,因為v所指向地址的值發(fā)生了改變。
答案:程序崩潰。因為GetMemory并不能傳遞動態(tài)內(nèi)存,Test函數(shù)中的str一直都是
NULL。
轉(zhuǎn)載于:https://www.cnblogs.com/yihujiu/p/6368077.html
總結(jié)
以上是生活随笔為你收集整理的内存泄露一个经典例子的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: LeetCode Construct t
- 下一篇: 元数据概述