kokodudu的專欄之malloc函式的用法
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 *),但是最好還是在前面進行強制型別轉換,因為這樣可以躲過一
些編譯器的檢查。
二、malloc()到底從哪裡得來了記憶體空間:
1、malloc()到底從哪裡得到了記憶體空間?答案是從堆裡面獲得空間。也就是說函式返回的指標是指向堆裡面的一塊記憶體。作業系統中有一個記錄空閒記憶體地址的連結串列。當作業系統收到程式的申請時,就會遍歷該連結串列,然後就尋找第一個空間大於所申請空間的堆結點,然後就將該結點從空閒結點連結串列中刪除,並將該結點的空間分配給程式。就是這樣!
說到這裡,不得不另外插入一個小話題,相信大家也知道是什麼話題了。什麼是堆?說到堆,又忍不住說到了棧!什麼是棧?下面就另外開個小部分專門而又簡單地說一下這個題外話:
2、什麼是堆:堆是大家共有的空間,分全域性堆和區域性堆。全域性堆就是所有沒有分配的空間,區域性堆就是使用者分配的空間。堆在作業系統對程序 初始化的時候分配,執行過程中也可以向系統要額外的堆,但是記得用完了要還給作業系統,要不然就是記憶體洩漏。
什麼是棧:棧是執行緒獨有的,儲存其執行狀態和區域性自動變數的。棧線上程開始的時候初始化,每個執行緒的棧互相獨立。每個函式都有自己的棧,棧被用來在函式之間傳遞引數。作業系統在切換執行緒的時候會自動的切換棧,就是切換SS/ESP暫存器。棧空間不需要在高階語言裡面顯式的分配和釋放。
以上的概念描述是標準的描述,不過有個別語句被我刪除,不知道因為這樣而變得不標準了^_^.
通過上面對概念的描述,可以知道:
棧是由編譯器自動分配釋放,存放函式的引數值、區域性變數的值等。操作方式類似於資料結構中的棧。
堆一般由程式設計師分配釋放,若不釋放,程式結束時可能由OS回收。注意這裡說是可能,並非一定。所以我想再強調一次,記得要釋放!
注意它與資料結構中的堆是兩回事,分配方式倒是類似於連結串列。(這點我上面稍微提過)
所以,舉個例子,如果你在函式上面定義了一個指標變數,然後在這個函式裡申請了一塊記憶體讓指標指向它。實際上,這個指標的地址是在棧上,但是它所指向的內容卻是在堆上面的!這一點要注意!所以,再想想,在一個函式裡申請了空間後,比如說下面這個函式:
// code...
void Function(void)
{
char *p = (char *)malloc(100 * sizeof(char));
}
就這個例子,千萬不要認為函式返回,函式所在的棧被銷燬指標也跟著銷燬,申請的記憶體也就一樣跟著銷燬了!這絕對是錯誤的!因為申請的記憶體在堆上,而函式所在的棧被銷燬跟堆完全沒有啥關係。所以,還是那句話:記得釋放!
3、free()到底釋放了什麼
這個問題比較簡單,其實我是想和第二大部分的題目相呼應而已!哈哈!free()釋放的是指標指向的記憶體!注意!釋放的是記憶體,不是指標!這點非常非常重要!指標是一個變數,只有程式結束時才被銷燬。釋放了記憶體空間後,原來指向這塊空間的指標還是存在!只不過現在指標指向的內容的垃圾,是未定義的,所以說是垃圾。因此,前面我已經說過了,釋放記憶體後把指標指向NULL,防止指標在後面不小心又被解引用了。非常重要啊這一點!
好了!這個“題外話”終於說完了。就這麼簡單說一次,知道個大概就可以了!下面就進入第三個部分:
三、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()的原始碼倒是容易簡單很多。只是有個疑問,下面指出!
// code...
void free(void *ptr)
{
struct mem_control_block *free;
free = ptr - sizeof(struct mem_control_block);
free->is_available = 1;
return;
}
看一下函式第二句,這句非常重要和關鍵。其實這句就是把指向可用空間的指標倒回去,讓它指向管理資訊的那塊空間,因為這裡是在值上減去了一個結構體的大小!後面那一句free->is_available = 1;我有點納悶!我的想法是:這裡is_available應該只是一個標記而已!因為從這個變數的名稱上來看,is_available 翻譯過來就是“是可以用”。不要說我土!我覺得變數名字可以反映一個變數的作用,特別是嚴謹的程式碼。這是原始碼,所以我覺得絕對是嚴謹的!!這個變數的值是1,表明是可以用的空間!只是這裡我想了想,如果把它改為0或者是其他值不知道會發生什麼事?!但是有一點我可以肯定,就是釋放絕對不會那麼順利進行!因為這是一個標記!
當然,這裡可能還是有人會有疑問,為什麼這樣就可以釋放呢??我剛才也有這個疑問。後來我想到,釋放是作業系統的事,那麼就free()這個原始碼來看,什麼也沒有釋放,對吧?但是它確實是確定了管理資訊的那塊記憶體的內容。所以,free()只是記錄了一些資訊,然後告訴作業系統那塊記憶體可以去釋放,具體怎麼告訴作業系統的我不清楚,但我覺得這個已經超出了我這篇文章的討論範圍了。
那麼,我之前有個錯誤的認識,就是認為指向那塊記憶體的指標不管移到那塊記憶體中的哪個位置都可以釋放那塊記憶體!但是,這是大錯特錯!釋放是不可以釋放一部分的!首先這點應該要明白。而且,從free()的原始碼看,ptr只能指向可用空間的首地址,不然,減去結構體大小之後一定不是指向管理資訊空間的首地址。所以,要確保指標指向可用空間的首地址!不信嗎?自己可以寫一個程式然後移動指向可用空間的指標,看程式會有會崩!
最後可能想到malloc()的原始碼看看malloc()到底是怎麼分配空間的,這裡面涉及到很多其他方面的知識!
相關推薦
kokodudu的專欄之malloc函式的用法
一、malloc()和free()的基本概念以及基本用法: 1、函式原型及說明: void *malloc(long NumBytes):該函式分配了NumBytes個位元組,並返回了指向這塊記憶體的指標。如果分配失敗,則返回一個空指標(NULL)。 關於分配失敗的原
C函式之memcpy 函式用法
分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!  
linux之select函式用法詳解
分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!  
C函式之memcpy 函式用法
標頭檔案:#include <string.h>memcpy() 用來複制記憶體,其原型為: void * memcpy ( void * dest, const void * src, size_t num );memcpy() 會複製 src 所指的記憶體內容的前 num 個位元組到 d
I/O複用模型之select函式用法——伺服器開發
現在我們介紹另外一種常用併發伺服器開發的技術——select函式I/O複用模型。 先來介紹select及相關的函式: select函式的作用是監聽指定的多個I/O的檔案描述符,在設定的時間內阻塞,當有一個或者多個I/O埠滿足某個“讀”或者“寫”的條件,則在fd_set型別
C函式之memcpy()函式用法
函式原型 void *memcpy(void*dest, const void *src, size_t n); 功能 由src指向地址為起始地址的連續n個位元組的資料複製到以destin指向地址為起始地址的空間內。 標頭檔案 #include<
malloc函式的用法
malloc函式是一種分配長度為num_bytes位元組的記憶體塊的函式,可以向系統申請分配指定size個位元組的記憶體空間。malloc的全稱是memory allocation(動態記憶體分配),當無法知道記憶體具體位置的時候,想要繫結真正的記憶體空間,就需要用到動態的分配記憶體。返回型別是 v
【 MATLAB 】nextpow2 函式用法之 Optimize FFT with Padding
您可以使用nextpow2來填充傳遞給fft的訊號。 這樣做可以在訊號長度不是2的精確冪次時加速FFT的計算。 Optimize FFT with Padding 下面這個例子展示了 使用填充優化FFT的案例,通過使用函式nextpow2完成: clc c
linux C函式之access函式的用法【轉】
1.函式功能: 檢查呼叫程序是否可以對指定的檔案執行某種操作。 2.函式原型: 1)函式標頭檔案 #include <stdio.h> #include <unistd.h> 2)函式 int access(
python基礎之zip()函式的用法
zip函式的用法: 詳情參考:菜鳥教程http://www.runoob.com/python/python-func-zip.html 注意!!!!!!!!!!!!!python2,python3的用法一樣但是返回值不一樣,python2返回的是一個列表,python3返回
Tensorflow函式用法之tf.argmax
tf.argmax(vector, 1):返回的是vector中的最大值的索引號,如果vector是一個向量,那就返回一個值,如果是一個矩陣,那就返回一個向量,這個向量的每一個維度都是相對應矩陣行的最大值元素的索引號。 import tensorflow as tf imp
三、PYTHON 學習筆記之 join 和 split 函式用法
python join 和 split方法的使用,join用來連線字串,split恰好相反,拆分字串的。 1.join用法示例 >>>li = ['my','name','is
C和指標之動態記憶體分配之(編寫calloc函式,函式內部使用malloc函式來獲取記憶體)
1、問題 編寫calloc函式,函式內部使用malloc函式來獲取記憶體 2、程式碼實現 #include <stdio.h> #include <stdlib.h&
Oracle之translate及replace函式用法總結
1、translate的用法 作用:translate的主要作用是提取,替換字串,其作用有時候和replace差不多。 語法:TRANSLATE(char, from_string, to_s
C++ STL 之 next_permutation 的用法(下一個序列函式,按字典序排)
這是一個求一個排序的下一個排列的函式,可以遍歷全排列,要包含標頭檔案<algorithm> 下面介紹一下next_permutation函式的用法 與之完全相反的函式還有prev_permutation (1)int型別的next_permuta
mysql函式之coalesce的用法
語法 COALESCE(value,…) Returns the first non-NULL value in the list, or NULL if there are no non-NULL values. The return type of C
前端踩坑小結:多個非同步請求在同一個函式裡面執行時的同步問題之promise的用法。
今天用VUE編寫專案時,涉及到兩個非同步請求在一個方法裡面對同一個變數進行操作,之前自己都沒意識到多個非同步請求對同一變數進行操作的時候會導致資料錯誤,結果今天除錯了半天,才想到了這個問題。也是怪自己還是一個新手,對於這種常識性的錯誤都不敏感。 於是自己查了一下相關資料,發
python學習番外篇之print輸出函式用法及原理總結
print輸出函式用法及原理總結: 在python2.x中,print作為關鍵字使用,輸出用print語句輸出,例如,x=5 ; print x ,但在python3.x中,print則成為了一個函式,輸出用print()函式輸出,例如:x=5 ; print(
c++ 標準模板庫 STL 演算法之 for_each 函式的使用用法詳解
std::for_each template <class InputIterator, class Function> Function for_each (InputIterator first, InputIterator last, Functi
【12】動態記憶體之malloc、calloc、realloc、free函式
建立動態記憶體:int *p = (int *) malloc(n*sizeof(int)); #include<stdio.h> #include<stdlib.h> #include<assert.h> // #include