當前位置

首頁 > 商務英語 > 計算機英語 > c語言中free的用法如何工作

c語言中free的用法如何工作

推薦人: 來源: 閱讀: 1.47W 次

free()與malloc()函數配對使用,釋放malloc函數申請的動態內存。下面本站小編就跟你們詳細介紹下c語言中free的用法,希望對你們有用。

c語言中free的用法如何工作
  c語言中free的用法:malloc()和free()

1、函數原型及說明:

void *malloc(long NumBytes):該函數分配了NumBytes個字節,並返回了指向這塊內存的指針。如果分配失敗,則返回一個空指針(NULL)。

關於分配失敗的原因,應該有多種,比如說空間不足就是一種。

void free(void *FirstByte): 該函數是將之前用malloc分配的空間還給程序或者是操作系統,也就是釋放了這塊內存,讓它重新得到自由

2、函數的用法:

其實這兩個函數用起來倒不是很難,也就是malloc()之後覺得用夠了就甩了它把它給free()了,舉個簡單例子:

程序代碼:

// Code...

char *Ptr = NULL;

Ptr = (char *)malloc(100 * sizeof(char));

if (NULL == Ptr)

{

exit (1);

}

gets(Ptr);

// code...

free(Ptr);

Ptr = NULL;

// code...

就是這樣!當然,具體情況要具體分析以及具體解決。比如說,你定義了一個指針,在一個函數裏申請了一塊內存然後通過函數返回傳遞給這個指針,那麼也許釋放這塊內存這項工作就應該留給其他函數了。

3、關於函數使用需要注意的一些地方:

A、申請了內存空間後,必須檢查是否分配成功。

B、當不需要再使用申請的內存時,記得釋放;釋放後應該把指向這塊內存的指針指向NULL,防止程序後面不小心使用了它。

C、這兩個函數應該是配對。如果申請後不釋放就是內存泄露;如果無故釋放那就是什麼也沒有做。釋放只能一次,如果釋放兩次及兩次以上會

出現錯誤(釋放空指針例外,釋放空指針其實也等於啥也沒做,所以釋放空指針釋放多少次都沒有問題)。

D、雖然malloc()函數的類型是(void *),任何類型的指針都可以轉換成(void *),但是最好還是在前面進行強制類型轉換,因爲這樣可以躲過一

些編譯器的檢查。

好了!最基礎的東西大概這麼說!現在進入第二部分:

  c語言中free的用法:malloc()到底從哪裏得來了內存空間

1、malloc()到底從哪裏得到了內存空間?答案是從堆裏面獲得空間。也就是說函數返回的指針是指向堆裏面的一塊內存。操作系統中有一個記錄空閒內存地址的鏈表。當操作系統收到程序的申請時,就會遍歷該鏈表,然後就尋找第一個空間大於所申請空間的堆結點,然後就將該結點從空閒結點鏈表中刪除,並將該結點的空間分配給程序。就是這樣!

說到這裏,不得不另外插入一個小話題,相信大家也知道是什麼話題了。什麼是堆?說到堆,又忍不住說到了棧!什麼是棧?下面就另外開個小部分專門而又簡單地說一下這個題外話:

2、什麼是堆:堆是大家共有的空間,分全局堆和局部堆。全局堆就是所有沒有分配的空間,局部堆就是用戶分配的空間。堆在操作系統對進程 初始化的時候分配,運行過程中也可以向系統要額外的堆,但是記得用完了要還給操作系統,要不然就是內存泄漏。

什麼是棧:棧是線程獨有的,保存其運行狀態和局部自動變量的。棧在線程開始的時候初始化,每個線程的棧互相獨立。每個函數都有自己的棧,棧被用來在函數之間傳遞參數。操作系統在切換線程的時候會自動的切換棧,就是切換SS/ESP寄存器。棧空間不需要在高級語言裏面顯式的分配和釋放。

以上的概念描述是標準的描述,不過有個別語句被我刪除,不知道因爲這樣而變得不標準了^_^.

通過上面對概念的描述,可以知道:

棧是由編譯器自動分配釋放,存放函數的參數值、局部變量的值等。操作方式類似於數據結構中的棧。

堆一般由程序員分配釋放,若不釋放,程序結束時可能由OS回收。注意這裏說是可能,並非一定。所以我想再強調一次,記得要釋放!

注意它與數據結構中的堆是兩回事,分配方式倒是類似於鏈表。(這點我上面稍微提過)值得關注,windows會用到系統的堆嗎。

所以,舉個例子,如果你在函數上面定義了一個指針變量,然後在這個函數裏申請了一塊內存讓指針指向它。實際上,這個指針的地址是在棧上,但是它所指向的內容卻是在堆上面的!這一點要注意!所以,再想想,在一個函數裏申請了空間後,比如說下面這個函數:

程序代碼:

// code...

void Function(void)

{

char *p = (char *)malloc(100 * sizeof(char));

}

就這個例子,千萬不要認爲函數返回,函數所在的棧被銷燬指針也跟着銷燬,申請的內存也就一樣跟着銷燬了!這絕對是錯誤的!因爲申請的內存在堆上,而函數所在的棧被銷燬跟堆完全沒有啥關係。所以,還是那句話:記得釋放!

3、free()到底釋放了什麼

這個問題比較簡單,其實我是想和第二大部分的題目相呼應而已!哈哈!free()釋放的是指針指向的內存!注意!釋放的是內存,不是指針!這點非常非常重要!指針是一個變量,只有程序結束時才被銷燬。釋放了內存空間後,原來指向這塊空間的指針還是存在!只不過現在指針指向的內容的垃圾,是未定義的,所以說是垃圾。因此,前面我已經說過了,釋放內存後把指針指向NULL,防止指針在後面不小心又被解引用了。非常重要啊這一點!

好了!這個“題外話”終於說完了。就這麼簡單說一次,知道個大概就可以了!下面就進入第三個部分:

  c語言中free的用法:malloc()以及free()的機制

這個部分我今天才有了新的認識!而且是轉折性的認識!所以,這部分可能會有更多一些認識上的錯誤!不對的地方請大家幫忙指出!

事實上,仔細看一下free()的函數原型,也許也會發現似乎很神奇,free()函數非常簡單,只有一個參數,只要把指向申請空間的指針傳遞

給free()中的參數就可以完成釋放工作!這裏要追蹤到malloc()的申請問題了。申請的時候實際上佔用的內存要比申請的大。因爲超出的空間是用來記錄對這塊內存的管理信息。先看一下在《UNIX環境高級編程》中第七章的一段話:

大多數實現所分配的存儲空間比所要求的要稍大一些,額外的空間用來記錄管理信息——分配塊的長度,指向下一個分配塊的指針等等。這就意味着如果寫過一個已分配區的尾端,則會改寫後一塊的管理信息。這種類型的錯誤是災難性的,但是因爲這種錯誤不會很快就暴露出來,所以也就很難發現。將指向分配塊的指針向後移動也可能會改寫本塊的管理信息。

以上這段話已經給了我們一些信息了。malloc()申請的空間實際我覺得就是分了兩個不同性質的空間。一個就是用來記錄管理信息的空間,另外一個就是可用空間了。而用來記錄管理信息的實際上是一個結構體。在C語言中,用結構體來記錄同一個對象的不同信息是

天經地義的事!下面看看這個結構體的原型:

程序代碼:

struct mem_control_block {

int is_available; //這是一個標記?

int size; //這是實際空間的大小

};

對於size,這個是實際空間大小。這裏其實我有個疑問,is_available是否是一個標記?因爲我看了free()的源代碼之後對這個變量感覺有點納悶(源代碼在下面分析)。這裏還請大家指出!

所以,free()就是根據這個結構體的信息來釋放malloc()申請的空間!而結構體的兩個成員的大小我想應該是操作系統的事了。但是這裏有一個問題,malloc()申請空間後返回一個指針應該是指向第二種空間,也就是可用空間!不然,如果指向管理信息空間的話,寫入的內容和結構體的類型有可能不一致,或者會把管理信息屏蔽掉,那就沒法釋放內存空間了,所以會發生錯誤!(感覺自己這裏說的是廢話)

好了!下面看看free()的源代碼,我自己分析了一下,覺得比起malloc()的源代碼倒是容易簡單很多。只是有個疑問,下面指出!

程序代碼:

看一下函數第二句,這句非常重要和關鍵。其實這句就是把指向可用空間的指針倒回去,讓它指向管理信息的那塊空間,因爲這裏是在值上減去了一個結構體的大小!後面那一句free->is_available =1;我有點納悶!我的想法是:這裏is_available應該只是一個標記而已!因爲從這個變量的名稱上來看,is_available翻譯過來就是“是可以用”。不要說我土!我覺得變量名字可以反映一個變量的作用,特別是嚴謹的代碼。這是源代碼,所以我覺得絕對是嚴謹的!!這個變量的值是1,表明是可以用的空間!只是這裏我想了想,如果把它改爲0或者是其他值不知道會發生什麼事?!但是有一點我可以肯定,就是釋放絕對不會那麼順利進行!因爲這是一個標記!

當然,這裏可能還是有人會有疑問,爲什麼這樣就可以釋放呢??我剛纔也有這個疑問。後來我想到,釋放是操作系統的事,那麼就free()這個源代碼來看,什麼也沒有釋放,對吧?但是它確實是確定了管理信息的那塊內存的內容。所以,free()只是記錄了一些信息,然後告訴操作系統那塊內存可以去釋放,具體怎麼告訴操作系統的我不清楚,但我覺得這個已經超出了我這篇文章的討論範圍了。

那麼,我之前有個錯誤的認識,就是認爲指向那塊內存的指針不管移到那塊內存中的哪個位置都可以釋放那塊內存!但是,這是大錯特錯!釋放是不可以釋放一部分的!首先這點應該要明白。而且,從free()的源代碼看,ptr只能指向可用空間的首地址,不然,減去結構體大小之後一定不是指向管理信息空間的首地址。所以,要確保指針指向可用空間的首地址!不信嗎?自己可以寫一個程序然後移動指向可用空間的指針,看程序會有會崩!

最後可能想到malloc()的源代碼看看malloc()到底是怎麼分配空間的,這裏面涉及到很多其他方面的知識!有興趣的朋友可以自己去下載源

代碼去看看。

=================================================

C語言的malloc分配的的內存大小

沒讀過malloc()的源碼,所以這裏純粹是"理論研究"。

malloc()在運行期動態分配分配內存,free()釋放由其分配的內存。malloc()在分配用戶傳入的大小的時候,還分配的一個相關的用於管理的額外內存,不過,用戶是看不到的。所以,

實際的大小 = 管理空間 + 用戶空間

那麼,這個管理內存放在什麼位置呢,它要讓free()函數能夠找到,這樣才能知道有多少內存要釋放,所以一種可能的方案是在分配內存的初始部分用若干個字節來存儲分配的內存的大小。這裏要注意一個問題,就是,在malloc()將這個分配的空間返回給某個指針後,這個指針的使用與其它指針應該是沒有差別的,所以,管理空間應該在這個指針指向的空間之外,但又要free()從這個指針可以找到管理信息,所以,這個管理空間的大小放在指針指向的相反方向。故malloc()的具體操作應該就是分配一塊內存,在前面若干字節中寫入管理信息,然後返回管理信息所佔字節之後的地址指針。

=================================================

malloc()工作機制

malloc函數的實質體現在,它有一個將可用的內存塊連接爲一個長長的列表的所謂空閒鏈表。調用malloc函數時,它沿連接表尋找一個大到足以滿足用戶請求所需要的內存塊。然後,將該內存塊一分爲二(一塊的大小與用戶請求的大小相等,另一塊的大小就是剩下的字節)。接下來,將分配給用戶的那塊內存傳給用戶,並將剩下的那塊(如果有的話)返回到連接表上。調用free函數時,它將用戶釋放的內存塊連接到空閒鏈上。到最後,空閒鏈會被切成很多的小內存片段,如果這時用戶申請一個大的內存片段,那麼空閒鏈上可能沒有可以滿足用戶要求的片段了。於是,malloc函數請求延時,並開始在空閒鏈上翻箱倒櫃地檢查各內存片段,對它們進行整理,將相鄰的小空閒塊合併成較大的內存塊。

malloc()在操作系統中的實現

在 C 程序中,多次使用malloc () 和 free()。不過,您可能沒有用一些時間去思考它們在您的操作系統中是如何實現的。本節將向您展示 malloc 和 free 的一個最簡化實現的代碼,來幫助說明管理內存時都涉及到了哪些事情。

在大部分操作系統中,內存分配由以下兩個簡單的函數來處理:

void *malloc (long numbytes):該函數負責分配 numbytes 大小的內存,並返回指向第一個字節的指針。

void free(void *firstbyte):如果給定一個由先前的 malloc 返回的指針,那麼該函數會將分配的空間歸還給進程的“空閒空間”。

malloc_init 將是初始化內存分配程序的函數。它要完成以下三件事:將分配程序標識爲已經初始化,找到系統中最後一個有效內存地址,然後建立起指向我們管理的內存的指針。這三個變量都是全局變量:

//清單 1. 我們的簡單分配程序的全局變量

如前所述,被映射的內存的邊界(最後一個有效地址)常被稱爲系統中斷點或者 當前中斷點。在很多 UNIX?系統中,爲了指出當前系統中斷點,必須使用 sbrk(0) 函數。 sbrk根據參數中給出的字節數移動當前系統中斷點,然後返回新的系統中斷點。使用參數 0 只是返回當前中斷點。這裏是我們的 malloc初始化代碼,它將找到當前中斷點並初始化我們的變量:

清單 2. 分配程序初始化函數

現在,爲了完全地管理內存,我們需要能夠追蹤要分配和回收哪些內存。在對內存塊進行了 free調用之後,我們需要做的是諸如將它們標記爲未被使用的等事情,並且,在調用 malloc 時,我們要能夠定位未被使用的內存塊。因此, malloc返回的每塊內存的起始處首先要有這個結構:

//清單 3. 內存控制塊結構定義

現在,您可能會認爲當程序調用 malloc 時這會引發問題 ——它們如何知道這個結構?答案是它們不必知道;在返回指針之前,我們會將其移動到這個結構之後,把它隱藏起來。這使得返回的指針指向沒有用於任何其他用途的內存。那樣,從調用程序的角度來看,它們所得到的全部是空閒的、開放的內存。然後,當通過 free()將該指針傳遞回來時,我們只需要倒退幾個內存字節就可以再次找到這個結構。

在討論分配內存之前,我們將先討論釋放,因爲它更簡單。爲了釋放內存,我們必須要做的惟一一件事情就是,獲得我們給出的指針,回退 sizeof(struct mem_control_block) 個字節,並將其標記爲可用的。這裏是對應的代碼:

清單 4. 解除分配函數

這就是我們的內存管理器。現在,我們只需要構建它,並在程序中使用它即可.多次調用malloc()後空閒內存被切成很多的小內存片段,這就使得用戶在申請內存使用時,由於找不到足夠大的內存空間,malloc()需要進行內存整理,使得函數的性能越來越低。聰明的程序員通過總是分配大小爲2的冪的內存塊,而最大限度地降低潛在的malloc性能喪失。也就是說,所分配的內存塊大小爲4字節、8字節、16字節、18446744073709551616字節,等等。這樣做最大限度地減少了進入空閒鏈的怪異片段(各種尺寸的小片段都有)的數量。儘管看起來這好像浪費了空間,但也容易看出浪費的空間永遠不會超過50%


猜你喜歡:

1.新手如何學習c語言

2.c語言考試學習的記憶方法

3.c語言程序設計怎麼學習

4.c語言實習心得

5.c語言實習心得體會