C語(yǔ)言之malloc動(dòng)態(tài)分配內(nèi)存和free釋放
malloc動(dòng)態(tài)分配內(nèi)存和free釋放
先看一個(gè)例子
#include<stdio.h> int max=10; //data區(qū),不產(chǎn)生具體的可執(zhí)行代碼。 void main() { //局部變量都在棧區(qū)。 棧內(nèi)存自動(dòng)分配,釋放。堆需要手動(dòng)malloc,free int a=10; //mov dword ptr[a], 0x0a; 在函數(shù)體內(nèi),產(chǎn)生具體的可執(zhí)行代碼。 }
malloc動(dòng)態(tài)分配的內(nèi)存在堆區(qū),其空間并不連續(xù)。函數(shù)返回的指針是指向堆里面的一塊內(nèi)存。操作系統(tǒng)中有一個(gè)記錄空閑內(nèi)存地址的鏈表。
當(dāng)操作系統(tǒng)收到程序的申請(qǐng)時(shí),就會(huì)遍歷該鏈表,然后就尋找第一個(gè)空間大于所申請(qǐng)空間的堆結(jié)點(diǎn),然后就將該結(jié)點(diǎn)從空閑結(jié)點(diǎn)鏈表中刪除,并將該結(jié)點(diǎn)的空間分配給程序。
什么是堆?說(shuō)到堆,又忍不住說(shuō)到了棧!什么是棧?
下面就另外開(kāi)個(gè)小部分專門(mén)而又簡(jiǎn)單地說(shuō)一下這個(gè)題外話:
- 什么是堆:
堆是大家共有的空間,分全局堆和局部堆。全局堆就是所有沒(méi)有分配的空間,局部堆就是用戶分配的空間。
堆在操作系統(tǒng)對(duì)進(jìn)程 初始化的時(shí)候分配,運(yùn)行過(guò)程中也可以向系統(tǒng)要額外的堆,但是記得用完了要還給操作系統(tǒng),要不然就是內(nèi)存泄漏。
- 什么是棧:
棧是線程獨(dú)有的,保存其運(yùn)行狀態(tài)和局部自動(dòng)變量的。棧在線程開(kāi)始的時(shí)候初始化,每個(gè)線程的?;ハ嗒?dú)立。每個(gè)函數(shù)都有自己的棧,棧被用來(lái)在函數(shù)之間傳遞參數(shù)。
操作系統(tǒng)在切換線程的時(shí)候會(huì)自動(dòng)的切換棧,就是切換SS/ESP寄存器。??臻g不需要在高級(jí)語(yǔ)言里面顯式的分配和釋放。
通過(guò)上面對(duì)概念的描述,可以知道:
棧是由編譯器自動(dòng)分配釋放,存放函數(shù)的參數(shù)值、局部變量的值等。操作方式類似于數(shù)據(jù)結(jié)構(gòu)中的棧。堆一般由程序員分配釋放,若不釋放,程序結(jié)束時(shí)可能由OS回收。
注意這里說(shuō)是可能,并非一定。所以我想再?gòu)?qiáng)調(diào)一次,記得要釋放!注意它與數(shù)據(jù)結(jié)構(gòu)中的堆是兩回事,分配方式倒是類似于鏈表。
所以,舉個(gè)例子,如果你在函數(shù)上面定義了一個(gè)指針變量,然后在這個(gè)函數(shù)里申請(qǐng)了一塊內(nèi)存讓指針指向它。
實(shí)際上,這個(gè)指針的地址是在棧上,但是它所指向的內(nèi)容卻是在堆上面的!這一點(diǎn)要注意!所以,再想想,在一個(gè)函數(shù)里申請(qǐng)了空間后,比如說(shuō)下面這個(gè)函數(shù):
// code... void Function(void) { char *p = (char *)malloc(100 * sizeof(char)); }
就這個(gè)例子,千萬(wàn)不要認(rèn)為函數(shù)返回,函數(shù)所在的棧被銷毀指針也跟著銷毀,申請(qǐng)的內(nèi)存也就一樣跟著銷毀了!這絕對(duì)是錯(cuò)誤的!因?yàn)樯暾?qǐng)的內(nèi)存在堆上,而函數(shù)所在的棧被銷毀跟堆完全沒(méi)有啥關(guān)系。所以,還是那句話:記得釋放!
1. 函數(shù)原型及說(shuō)明
void *malloc(long NumBytes)
:該函數(shù)分配了NumBytes個(gè)字節(jié),并返回了指向這塊內(nèi)存的指針。如果分配失敗,則返回一個(gè)空指針(NULL)。關(guān)于分配失敗的原因,應(yīng)該有多種,比如說(shuō)空間不足就是一種。void free(void *FirstByte)
: 該函數(shù)是將之前用malloc分配的空間還給程序或者是操作系統(tǒng),也就是釋放了這塊內(nèi)存,讓它重新得到自由。
2. 關(guān)于函數(shù)使用需要注意的地方
A、申請(qǐng)了內(nèi)存空間后
必須檢查是否分配成功。
B、當(dāng)不需要再使用申請(qǐng)的內(nèi)存時(shí)
記得釋放;釋放后應(yīng)該把指向這塊內(nèi)存的指針指向NULL,防止程序后面不小心使用了它。
#include<stdio.h> #include<malloc.h> void main() { int n; int *p=NULL; scanf("%d",&n); p=(int *)malloc(sizeof(int)*n); //n=10 for(int i=0;i<n;++i) { p[i]=i; } for(int j=0;j<n;++j) { printf("%d\n",p[j]); } free(p); p=NULL; //free釋放堆空間后,必須把無(wú)效指針變?yōu)榭铡#ǚ质忠蹇涨叭蔚囊磺校? }
如何表示這一塊堆內(nèi)存已經(jīng)被占用,避免別的程序又來(lái)分配我的空間呢。用標(biāo)志位來(lái)標(biāo)記,所以加上了標(biāo)志位之后n=10時(shí)堆中開(kāi)辟的空間肯定不止40字節(jié)。
注意 free釋放的是指針指向的內(nèi)存,不是指針。指針只是一個(gè)變量,只有程序結(jié)束時(shí)才被銷毀。釋放了內(nèi)存空間后,原來(lái)指向這塊空間的指針還是存在。
當(dāng)空間釋放(free)時(shí),標(biāo)志位狀態(tài)就會(huì)發(fā)生變化,此時(shí)又允許其他程序分配我原本的內(nèi)存空間。此時(shí)指針變?yōu)闊o(wú)效指針,必須把無(wú)效指針變?yōu)榭铡?/p>
否則失效指針仍可對(duì)內(nèi)存操作,但有可能這一塊內(nèi)存被分配給別的程序,就會(huì)使無(wú)效指針修改別人的數(shù)據(jù)。
麻煩大了。 如:
void main() { int a=10; int *p=NULL; p=(int *)malloc(sizeof(int)); *p=100; printf("%d \n",*p); free(p); //指針仍指向那塊,但系統(tǒng)已經(jīng)將這塊空間標(biāo)記為未用空間。 //失效指針。 printf("%d \n",*p); *p=200; printf("%d \n",*p); p=NULL; //第一次打印100 第二次打印隨機(jī)值 第三次打印200 }
C、這兩個(gè)函數(shù)應(yīng)該是配對(duì)
如果申請(qǐng)后不釋放就是內(nèi)存泄露;如果無(wú)故釋放那就是什么也沒(méi)有做。
釋放只能一次,如果釋放兩次及兩次以上會(huì)出現(xiàn)錯(cuò)誤(釋放空指針例外,釋放空指針其實(shí)也等于啥也沒(méi)做,所以釋放空指針釋放多少次都沒(méi)有問(wèn)題)。
如:
void main() { int a=10; int *p=(int *)malloc(sizeof(int)); *p=200; printf("%d \n",*p); free(p); printf("%d \n",*p); //p=NULL; 先置空再free,系統(tǒng)不出錯(cuò)。什么也不做。(本來(lái)就空指針去離婚,人不搭理你。) free(p); //系統(tǒng)報(bào)錯(cuò)。不能對(duì)同一個(gè)空間重復(fù)進(jìn)行釋放。(重復(fù)離婚,放狗咬人。) }
D、malloc申請(qǐng)的指針不要?jiǎng)铀?/h4>
可以動(dòng)它的值,但自身不能動(dòng)。
這里涉及malloc()以及free()的機(jī)制
“大多數(shù)實(shí)現(xiàn)所分配的存儲(chǔ)空間比所要求的要稍大一些,額外的空間用來(lái)記錄管理信息——分配塊的長(zhǎng)度,指向下一個(gè)分配塊的指針等等。這就意味著如果寫(xiě)過(guò)一個(gè)已分配區(qū)的尾端,則會(huì)改寫(xiě)后一塊的管理信息。這種類型的錯(cuò)誤是災(zāi)難性的,但是因?yàn)檫@種錯(cuò)誤不會(huì)很快就暴露出來(lái),所以也就很難發(fā)現(xiàn)。將指向分配塊的指針向后移動(dòng)也可能會(huì)改寫(xiě)本塊的管理信息。”
以上這段話已經(jīng)給了我們一些信息了。malloc()申請(qǐng)的空間實(shí)際我覺(jué)得就是分了兩個(gè)不同性質(zhì)的空間。一個(gè)就是用來(lái)記錄管理信息的空間,另外一個(gè)就是可用空間了。而用來(lái)記錄管理信息的實(shí)際上是一個(gè)結(jié)構(gòu)體。
在C語(yǔ)言中,用結(jié)構(gòu)體來(lái)記錄同一個(gè)對(duì)象的不同信息是天經(jīng)地義的事!
下面看看這個(gè)結(jié)構(gòu)體的原型:
struct mem_control_block { int is_available; //這是一個(gè)標(biāo)記 int size; //這是實(shí)際空間的大小 };
free()就是根據(jù)這個(gè)結(jié)構(gòu)體的信息來(lái)釋放malloc()申請(qǐng)的空間,malloc()申請(qǐng)空間后返回一個(gè)指針應(yīng)該是指向第二種空間,也就是可用空間。
看看free()的源代碼
void free(void *ptr) { struct mem_control_block *free; free = ptr - sizeof(struct mem_control_block); free->is_available = 1; return; }
看一下函數(shù)第二句,這句非常重要和關(guān)鍵。
其實(shí)這句就是把指向可用空間的指針倒回去,讓它指向管理信息的那塊空間,因?yàn)檫@里是在值上減去了一個(gè)結(jié)構(gòu)體的大小!
那么,我之前有個(gè)錯(cuò)誤的認(rèn)識(shí),就是認(rèn)為指向那塊內(nèi)存的指針不管移到那塊內(nèi)存中的哪個(gè)位置都可以釋放那塊內(nèi)存!但是,這是大錯(cuò)特錯(cuò)!釋放是不可以釋放一部分的!首先這點(diǎn)應(yīng)該要明白。
而且,從free()的源代碼看,ptr只能指向可用空間的首地址,不然,減去結(jié)構(gòu)體大小之后一定不是指向管理信息空間的首地址。
所以,要確保指針指向可用空間的首地址!不信嗎?自己可以寫(xiě)一個(gè)程序然后移動(dòng)指向可用空間的指針,看程序會(huì)不會(huì)崩!
#include<stdio.h> #include<malloc.h> void main() { int n; int *p=NULL; scanf("%d",&n); p=(int *)malloc(sizeof(int)*n); //此時(shí)p指向第一個(gè)空間元素 for(int i=0;i<n;i++) { *p=i; p+=1; } //此時(shí)p指向第n個(gè)元素。 free(p); //釋放時(shí)就找不到標(biāo)志位了。 p=NULL; }
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
快速學(xué)習(xí)C語(yǔ)言中for循環(huán)語(yǔ)句的基本使用方法
這篇文章主要簡(jiǎn)單介紹了C語(yǔ)言中for循環(huán)語(yǔ)句的基本使用方法,是C語(yǔ)言入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-11-11C語(yǔ)言實(shí)現(xiàn)文件讀寫(xiě)操作的幾種常用方法
C語(yǔ)言提供了一系列文件操作函數(shù),使得我們可以通過(guò)程序?qū)ξ募M(jìn)行讀寫(xiě)操作,本文主要介紹了C語(yǔ)言實(shí)現(xiàn)文件讀寫(xiě)操作的幾種常用方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03C++ 實(shí)現(xiàn)稀疏矩陣的壓縮存儲(chǔ)的實(shí)例
這篇文章主要介紹了C++ 實(shí)現(xiàn)稀疏矩陣的壓縮存儲(chǔ)的實(shí)例的相關(guān)資料,M*N的矩陣,矩陣中有效值的個(gè)數(shù)遠(yuǎn)小于無(wú)效值的個(gè)數(shù),且這些數(shù)據(jù)的分布沒(méi)有規(guī)律,需要的朋友可以參考下2017-07-07C語(yǔ)言實(shí)現(xiàn)學(xué)生成績(jī)管理系統(tǒng)課程設(shè)計(jì)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)學(xué)生成績(jī)管理系統(tǒng)課程設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07C語(yǔ)言操作符進(jìn)階教程(表達(dá)式求值隱式類型轉(zhuǎn)換方法)
這篇文章主要為大家介紹了C語(yǔ)言操作符進(jìn)階教程(表達(dá)式求值隱式類型轉(zhuǎn)換方法)2022-02-02