當(dāng)前位置:首頁 > 公眾號精選 > C語言與CPP編程
[導(dǎo)讀]定義了指針變量,但是沒有為指針分配內(nèi)存,即指針沒有指向一塊合法的內(nèi)存。淺顯的例子就不舉了,這里舉幾個比較隱蔽的例子。

來源:https://www.cnblogs.com/XDJjy/p/3954404.html

一、指針沒有指向一塊合法的內(nèi)存

定義了指針變量,但是沒有為指針分配內(nèi)存,即指針沒有指向一塊合法的內(nèi)存。淺顯的例子就不舉了,這里舉幾個比較隱蔽的例子。

1、結(jié)構(gòu)體成員指針未初始化

struct?student
{
???char?*name;
???int?score;
}stu,*pstu;

int?main()
{
???strcpy(stu.name,"Jimy");
???stu.score?=?99;
???return?0;
}

很多初學(xué)者犯了這個錯誤還不知道是怎么回事。這里定義了結(jié)構(gòu)體變量stu,但是他沒想到這個結(jié)構(gòu)體內(nèi)部char *name 這成員在定義結(jié)構(gòu)體變量stu 時,只是給name 這個指針變量本身分配了4 個字節(jié)。name 指針并沒有指向一個合法的地址,這時候其內(nèi)部存的只是一些亂碼。所以在調(diào)用strcpy 函數(shù)時,會將字符串"Jimy"往亂碼所指的內(nèi)存上拷貝,而這塊內(nèi)存name 指針根本就無權(quán)訪問,導(dǎo)致出錯。解決的辦法是為name 指針malloc 一塊空間。

同樣,也有人犯如下錯誤:

int?main()
{
???pstu?=?(struct?student*)malloc(sizeof(struct?student));
???strcpy(pstu->name,"Jimy");
???pstu->score?=?99;
???free(pstu);
???return?0;
}

為指針變量pstu 分配了內(nèi)存,但是同樣沒有給name 指針分配內(nèi)存。錯誤與上面第一種情況一樣,解決的辦法也一樣。這里用了一個malloc 給人一種錯覺,以為也給name 指針分配了內(nèi)存。

2、沒有為結(jié)構(gòu)體指針分配足夠的內(nèi)存

int?main()
{
???pstu?=?(struct?student*)malloc(sizeof(struct?student*));
???strcpy(pstu->name,"Jimy");
???pstu->score?=?99;
???free(pstu);
???return?0;
}

為pstu 分配內(nèi)存的時候,分配的內(nèi)存大小不合適。這里把sizeof(struct student)誤寫為sizeof(struct student*)。當(dāng)然name 指針同樣沒有被分配內(nèi)存。解決辦法同上。

3、函數(shù)的入口校驗

不管什么時候,我們使用指針之前一定要確保指針是有效的。

一般在函數(shù)入口處使用assert(NULL != p)對參數(shù)進行校驗。在非參數(shù)的地方使用if(NULL != p)來校驗。但這都有一個要求,即p 在定義的同時被初始化為NULL 了。比如上面的例子,即使用if(NULL != p)校驗也起不了作用,因為name 指針并沒有被初始化為NULL,其內(nèi)部是一個非NULL 的亂碼。

assert 是一個宏,而不是函數(shù),包含在assert.h 頭文件中。如果其后面括號里的值為假,則程序終止運行,并提示出錯;如果后面括號里的值為真,則繼續(xù)運行后面的代碼。這個宏只在Debug 版本上起作用,而在Release 版本被編譯器完全優(yōu)化掉,這樣就不會影響代碼的性能。

有人也許會問,既然在Release 版本被編譯器完全優(yōu)化掉,那Release 版本是不是就完全沒有這個參數(shù)入口校驗了呢?這樣的話那不就跟不使用它效果一樣嗎?

是的,使用assert 宏的地方在Release 版本里面確實沒有了這些校驗。但是我們要知道,assert 宏只是幫助我們調(diào)試代碼用的,它的一切作用就是讓我們盡可能的在調(diào)試函數(shù)的時候把錯誤排除掉,而不是等到Release 之后。它本身并沒有除錯功能。再有一點就是,參數(shù)出現(xiàn)錯誤并非本函數(shù)有問題,而是調(diào)用者傳過來的實參有問題。assert 宏可以幫助我們定位錯誤,而不是排除錯誤。

二、為指針分配的內(nèi)存太小

為指針分配了內(nèi)存,但是內(nèi)存大小不夠,導(dǎo)致出現(xiàn)越界錯誤。

char?*p1?=?“abcdefg”;
char?*p2?=?(char?*)malloc(sizeof(char)*strlen(p1));
strcpy(p2,p1);

p1 是字符串常量,其長度為7 個字符,但其所占內(nèi)存大小為8 個byte。初學(xué)者往往忘了字符串常量的結(jié)束標(biāo)志“\0”。這樣的話將導(dǎo)致p1 字符串中最后一個空字符“\0”沒有被拷貝到p2 中。解決的辦法是加上這個字符串結(jié)束標(biāo)志符:

char?*p2?=?(char?*)malloc(sizeof(char)*strlen(p1)+1*sizeof(char));

這里需要注意的是,只有字符串常量才有結(jié)束標(biāo)志符。比如下面這種寫法就沒有結(jié)束標(biāo)志符了:

char?a[7]?=?{‘a(chǎn)’,’b’,’c’,’d’,’e’,’f’,’g’};

另外,不要因為char 類型大小為1 個byte 就省略sizof(char)這種寫法。這樣只會使你的代碼可移植性下降。

三、內(nèi)存分配成功,但并未初始化

犯這個錯誤往往是由于沒有初始化的概念或者是以為內(nèi)存分配好之后其值自然為0。未初始化指針變量也許看起來不那么嚴(yán)重,但是它確確實實是個非常嚴(yán)重的問題,而且往往出現(xiàn)這種錯誤很難找到原因。

曾經(jīng)有一個學(xué)生在寫一個windows 程序時,想調(diào)用字庫的某個字體。而調(diào)用這個字庫需要填充一個結(jié)構(gòu)體。他很自然的定義了一個結(jié)構(gòu)體變量,然后把他想要的字庫代碼賦值給了相關(guān)的變量。但是,問題就來了,不管怎么調(diào)試,他所需要的這種字體效果總是不出來。我在檢查了他的代碼之后,沒有發(fā)現(xiàn)什么問題,于是單步調(diào)試。在觀察這個結(jié)構(gòu)體變量的內(nèi)存時,發(fā)現(xiàn)有幾個成員的值為亂碼。就是其中某一個亂碼惹得禍!因為系統(tǒng)會按照這個結(jié)構(gòu)體中的某些特定成員的值去字庫中尋找匹配的字體,當(dāng)這些值與字庫中某種字體的某些項匹配時,就調(diào)用這種字體。但是很不幸,正是因為這幾個亂碼,導(dǎo)致沒有找到相匹配的字體!因為系統(tǒng)并無法區(qū)分什么數(shù)據(jù)是亂碼,什么數(shù)據(jù)是有效的數(shù)據(jù)。只要有數(shù)據(jù),系統(tǒng)就理所當(dāng)然的認(rèn)為它是有效的。

也許這種嚴(yán)重的問題并不多見,但是也絕不能掉以輕心。所以在定義一個變量時,第一件事就是初始化。你可以把它初始化為一個有效的值,比如:

int i = 10;
char?*p?=?(char?*)malloc(sizeof(char));

但是往往這個時候我們還不確定這個變量的初值,這樣的話可以初始化為0 或NULL。

int?i?=?0;
char?*p?=?NULL;

如果定義的是數(shù)組的話,可以這樣初始化:

int?a[10]?=?{0};

或者用memset 函數(shù)來初始化為0:

memset(a,0,sizeof(a));

memset 函數(shù)有三個參數(shù),第一個是要被設(shè)置的內(nèi)存起始地址;第二個參數(shù)是要被設(shè)置的值;第三個參數(shù)是要被設(shè)置的內(nèi)存大小,單位為byte。這里并不想過多的討論memset 函數(shù)的用法,如果想了解更多,請參考相關(guān)資料。

至于指針變量如果未被初始化,會導(dǎo)致if 語句或assert 宏校驗失敗。這一點,上面已有分析。

四、內(nèi)存越界

內(nèi)存分配成功,且已經(jīng)初始化,但是操作越過了內(nèi)存的邊界。這種錯誤經(jīng)常是由于操作數(shù)組或指針時出現(xiàn)“多1”或“少1”。比如:

int?a[10]?=?{0};
for?(i=0;?i<=10;?i++)
{
???a[i]?=?i;
}

所以,for 循環(huán)的循環(huán)變量一定要使用半開半閉的區(qū)間,而且如果不是特殊情況,循環(huán)變量盡量從0 開始。

五、內(nèi)存泄漏

內(nèi)存泄漏幾乎是很難避免的,不管是老手還是新手,都存在這個問題。甚至包括windows,Linux 這類軟件,都或多或少有內(nèi)存泄漏。也許對于一般的應(yīng)用軟件來說,這個問題似乎不是那么突出,重啟一下也不會造成太大損失。但是如果你開發(fā)的是嵌入式系統(tǒng)軟件呢?比如汽車制動系統(tǒng),心臟起搏器等對安全要求非常高的系統(tǒng)。你總不能讓心臟起搏器重啟吧,人家閻王老爺是非常好客的。

會產(chǎn)生泄漏的內(nèi)存就是堆上的內(nèi)存(這里不討論資源或句柄等泄漏情況),也就是說由malloc 系列函數(shù)或new 操作符分配的內(nèi)存。如果用完之后沒有及時free 或delete,這塊內(nèi)存就無法釋放,直到整個程序終止。

1、告老還鄉(xiāng)求良田

怎么去理解這個內(nèi)存分配和釋放過程呢?先看下面這段對話:

萬歲爺:愛卿,你為朕立下了汗馬功勞,想要何賞賜???

某功臣:萬歲,黃金白銀,臣視之如糞土。臣年歲已老,欲告老還鄉(xiāng)。臣乞良田千畝以蔭后世,別無他求。

萬歲爺:愛卿,你勞苦功高,卻僅要如此小賞,朕今天就如你所愿。戶部劉侍郎,查看湖廣一帶是否還有千畝上等良田未曾封賞。

劉侍郎:長沙尚有五萬余畝上等良田未曾封賞。

萬歲爺:在長沙撥良田千畝封賞愛卿。愛卿,良田千畝,你欲何用?。?/p>

某功臣:謝萬歲。長沙一帶,適合種水稻,臣想用來種水稻。種水稻需要把田分為一畝一塊,方便耕種。

。。。。

2、如何使用malloc 函數(shù)

不要莫名其妙,其實上面這段小小的對話,就是malloc 的使用過程。malloc 是一個函數(shù),專門用來從堆上分配內(nèi)存。使用malloc 函數(shù)需要幾個要求:

內(nèi)存分配給誰?這里是把良田分配給某功臣。

分配多大內(nèi)存?這里是分配一千畝。

是否還有足夠內(nèi)存分配?這里是還有足夠良田分配。

內(nèi)存的將用來存儲什么格式的數(shù)據(jù),即內(nèi)存用來做什么?

這里是用來種水稻,需要把田分成一畝一塊。分配好的內(nèi)存在哪里?這里是在長沙。

如果這五點都確定,那內(nèi)存就能分配。下面先看malloc 函數(shù)的原型:?(void *)malloc(int size)?malloc 函數(shù)的返回值是一個void 類型的指針,參數(shù)為int 類型數(shù)據(jù),即申請分配的內(nèi)存大小,單位是byte。內(nèi)存分配成功之后,malloc 函數(shù)返回這塊內(nèi)存的首地址。你需要一個指針來接收這個地址。但是由于函數(shù)的返回值是void *類型的,所以必須強制轉(zhuǎn)換成你所接收的類型。也就是說,這塊內(nèi)存將要用來存儲什么類型的數(shù)據(jù)。比如:?char *p = (char *)malloc(100);?在堆上分配了100 個字節(jié)內(nèi)存,返回這塊內(nèi)存的首地址,把地址強制轉(zhuǎn)換成char *類型后賦給char *類型的指針變量p。同時告訴我們這塊內(nèi)存將用來存儲char 類型的數(shù)據(jù)。也就是說你只能通過指針變量p 來操作這塊內(nèi)存。這塊內(nèi)存本身并沒有名字,對它的訪問是匿名訪問。

上面就是使用malloc 函數(shù)成功分配一塊內(nèi)存的過程。但是,每次你都能分配成功嗎?

不一定。上面的對話,皇帝讓戶部侍郎查詢是否還有足夠的良田未被分配出去。使用malloc函數(shù)同樣要注意這點:如果所申請的內(nèi)存塊大于目前堆上剩余內(nèi)存塊(整塊),則內(nèi)存分配會失敗,函數(shù)返回NULL。注意這里說的“堆上剩余內(nèi)存塊”不是所有剩余內(nèi)存塊之和,因為malloc 函數(shù)申請的是連續(xù)的一塊內(nèi)存。

既然malloc 函數(shù)申請內(nèi)存有不成功的可能,那我們在使用指向這塊內(nèi)存的指針時,必須用if(NULL != p)語句來驗證內(nèi)存確實分配成功了。

3、用malloc 函數(shù)申請0 字節(jié)內(nèi)存

另外還有一個問題:用malloc 函數(shù)申請0 字節(jié)內(nèi)存會返回NULL 指針嗎?

可以測試一下,也可以去查找關(guān)于malloc 函數(shù)的說明文檔。申請0 字節(jié)內(nèi)存,函數(shù)并不返回NULL,而是返回一個正常的內(nèi)存地址。但是你卻無法使用這塊大小為0 的內(nèi)存。這好尺子上的某個刻度,刻度本身并沒有長度,只有某兩個刻度一起才能量出長度。對于這一點一定要小心,因為這時候if(NULL != p)語句校驗將不起作用。

4、內(nèi)存釋放

既然有分配,那就必須有釋放。不然的話,有限的內(nèi)存總會用光,而沒有釋放的內(nèi)存卻在空閑。與malloc 對應(yīng)的就是free 函數(shù)了。free 函數(shù)只有一個參數(shù),就是所要釋放的內(nèi)存塊的首地址。比如上例:?free(p);?free 函數(shù)看上去挺狠的,但它到底作了什么呢?其實它就做了一件事:斬斷指針變量與這塊內(nèi)存的關(guān)系。比如上面的例子,我們可以說malloc 函數(shù)分配的內(nèi)存塊是屬于p 的,因為我們對這塊內(nèi)存的訪問都需要通過p 來進行。free 函數(shù)就是把這塊內(nèi)存和p 之間的所有關(guān)系斬斷。從此p 和那塊內(nèi)存之間再無瓜葛。至于指針變量p 本身保存的地址并沒有改變,但是它對這個地址處的那塊內(nèi)存卻已經(jīng)沒有所有權(quán)了。那塊被釋放的內(nèi)存里面保存的值也沒有改變,只是再也沒有辦法使用了。

這就是free 函數(shù)的功能。按照上面的分析,如果對p 連續(xù)兩次以上使用free 函數(shù),肯定會發(fā)生錯誤。因為第一使用free 函數(shù)時,p 所屬的內(nèi)存已經(jīng)被釋放,第二次使用時已經(jīng)無內(nèi)存可釋放了。關(guān)于這點,我上課時讓學(xué)生記住的是:一定要一夫一妻制,不然肯定出錯。

malloc 兩次只free 一次會內(nèi)存泄漏;malloc 一次free 兩次肯定會出錯。也就是說,在程序中malloc 的使用次數(shù)一定要和free 相等,否則必有錯誤。這種錯誤主要發(fā)生在循環(huán)使用malloc 函數(shù)時,往往把malloc 和free 次數(shù)弄錯了。這里留個 練習(xí):

寫兩個函數(shù),一個生成鏈表,一個釋放鏈表。兩個函數(shù)的參數(shù)都只使用一個表頭指針。

5、內(nèi)存釋放之后

既然使用free 函數(shù)之后指針變量p 本身保存的地址并沒有改變,那我們就需要重新把p的值變?yōu)镹ULL:?p = NULL;?這個NULL 就是我們前面所說的“栓野狗的鏈子”。如果你不栓起來遲早會出問題的。比如:在free(p)之后,你用if(NULL != p)這樣的校驗語句還能起作用嗎?例如:

char?*p?=?(char?*)malloc(100);
strcpy(p,?“hello”);
free(p);?/*?p?所指的內(nèi)存被釋放,但是p?所指的地址仍然不變*/

if?(NULL?!=?p)
{
??/*?沒有起到防錯作用*/
??strcpy(p,?“world”);?/*?出錯*/
}

釋放完塊內(nèi)存之后,沒有把指針置NULL,這個指針就成為了“野指針”,也有書叫“懸垂指針”。這是很危險的,而且也是經(jīng)常出錯的地方。所以一定要記住一條:free 完之后,一定要給指針置NULL。

同時留一個問題:對NULL 指針連續(xù)free 多次會出錯嗎?為什么?如果讓你來設(shè)計free函數(shù),你會怎么處理這個問題?

六、內(nèi)存已經(jīng)被釋放了,但是繼續(xù)通過指針來使用

這里一般有三種情況:

第一種:就是上面所說的,free(p)之后,繼續(xù)通過p 指針來訪問內(nèi)存。解決的辦法就是給p 置NULL。

第二種:函數(shù)返回棧內(nèi)存。這是初學(xué)者最容易犯的錯誤。比如在函數(shù)內(nèi)部定義了一個數(shù)組,卻用return 語句返回指向該數(shù)組的指針。解決的辦法就是弄明白棧上變量的生命周期。

第三種:內(nèi)存使用太復(fù)雜,弄不清到底哪塊內(nèi)存被釋放,哪塊沒有被釋放。解決的辦法是重新設(shè)計程序,改善對象之間的調(diào)用關(guān)系。

上面詳細(xì)討論了常見的六種錯誤及解決對策,希望讀者仔細(xì)研讀,盡量使自己對每種錯誤發(fā)生的原因及預(yù)防手段爛熟于胸。一定要多練,多調(diào)試代碼,同時多總結(jié)經(jīng)驗。

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉