C 編程中,常常須要操做的內存可分爲下面幾個類別:程序員
C 標準函數庫提供了許多函數來實現對堆上內存管理,其中包括:malloc()函數,free()函數,calloc()函數和realloc()函數。使用這些函數須要包含頭文件stdlib.h。它們的聲明以下:編程
- void * malloc(size_t n);
- void free(void * p);
- void *calloc(size_t n, size_t size);
- void * realloc(void * p, size_t n);
malloc()函數能夠從堆上得到指定字節的內存空間,其函數聲明以下:數組
void * malloc(size_t n);安全
其中,形參n爲要求 分配的字節數。若是函數執行成功,malloc()返回得到內存空間的首地址;若是函數執行失敗,那麼返回值爲NULL。因爲 malloc()函數值的類型爲void型指針,所以,能夠將其值類型轉換後賦給任意類型指針,這樣就能夠經過操做該類型指針來操做從堆上得到的內存空間。數據結構
須要注意的是,malloc()函數分配獲得的內存空間是未初始化的。所以,通常在使用該內存空間時,要調用另外一個函數memset來將其初始化爲全0。memset函數的聲明以下:函數
void * memset (void * p, int c, size_t n);spa
該函數能夠將指定的內存空間按字節單位置爲指定的字符c。其中,p爲要清零的內存空間的首地址,c爲要設定的值,n爲被操做的內存空間的字節長度。若是要用memset清0,變量c實參要爲0。malloc()函數和memset函數的操做語句通常以下:操作系統
int *p = NULL; p = (int *) malloc(sizeof(int)); if (p == NULL) { printf("Can’t get memory!\n"); } memset(p, 0, sizeof(int));
注意:經過malloc()函數獲得的堆內存必須使用memset()函數來初始化。指針
示例代碼:htm
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { int *p = NULL; p = (int *) malloc(sizeof(int)); if (NULL == p) { printf("Can't get memory!\n"); return -1; } printf("%d\n", *p); // 輸出分配的空間上的值 memset(p, 0, sizeof(int)); // 將p指向的空間清0 printf("%d\n", *p); // 輸出調用memset函數後的結果 *p = 2; printf("%d\n", *p); return 0; }
從堆上得到的內存空間在程序結束之後,系統不會將其自動釋放,須要程序員來本身管理。一個程序結束時,必須保證全部從堆上得到的內存空間已被安全釋放,不然,會致使內存泄露。例如上面的demo就會發生內存泄露。
free()函數能夠實現釋放內存的功能。其函數聲明爲:
void free(void * p);
因爲形參爲void指針,free()函數能夠接受任意類型的指針實參。
可是,free()函數只是釋放指針指向的內容,而該指針仍然指向原來指向的地方,此時,指針爲野指針,若是此時操做該指針會致使不可預期的錯誤。安全作法 是:在使用free()函數釋放指針指向的空間以後,將指針的值置爲NULL。所以,對於上面的demo,須要在return
語句前加入如下兩行語句:
free(p); p = NULL;
注意:使用malloc()函數分配的堆空間在程序結束以前必須釋放。
calloc()函數的功能與malloc()函數的功能類似,都是從堆分配內存。其函數聲明以下:
void *calloc(size_t n, size_t size);
函數返回值爲void型指針。若是執行成功,函數從堆上得到 size X n 的字節空間,並返回該空間的首地址。若是執行失敗,函數返回NULL。該函數與malloc()函數的一個顯著不一樣時是,calloc()函數獲得的內存空間是通過初始化的,其內容全爲0。calloc()函數適合爲數組申請空間,能夠將size設置爲數組元素的空間長度,將n設置爲數組的容量。
示例代碼:
#include <stdio.h> #include <stdlib.h> #define SIZE 5 int main() { int *p = NULL; int i = 0; // 爲p從堆上分配SIZE個int型空間 p = (int *) calloc(SIZE, sizeof(int)); if (NULL == p) { printf("Error in calloc.\n"); return -1; } // 爲p指向的SIZE個int型空間賦值 for (i = 0; i < SIZE; i++) { p[i] = i; } // 輸出各個空間的值 for (i = 0; i < SIZE; i++) { printf("p[%d]=%d\n", i, p[i]); } free(p); p = NULL; return 0; }
提示:calloc()函數的分配的內存也須要自行釋放。
realloc()函數的功能比malloc()函數和calloc()函數的功能更爲豐富,能夠實現內存分配和內存釋放的功能,其函數聲明以下:
void * realloc(void * p, size_t n);
其中,指針p必須爲指向堆內存空間的指針,即由malloc()函數、calloc()函數或realloc()函數分配空間的指針。realloc()函數將指針 p指向的內存塊的大小改變爲n字節。若是n小於或等於p以前指向的空間大小,那麼。保持原有狀態不變。若是n大於原來p以前指向的空間大小,那麼,系統將 從新爲p從堆上分配一塊大小爲n的內存空間,同時,將原來指向空間的內容依次複製到新的內存空間上,p以前指向的空間被釋放。realloc()函數分配的空間也是未初始化的。
注意:使用malloc()函數,calloc()函數和realloc()函數分配的內存空間都要使用free()函數或指針參數爲NULL的realloc()函數來釋放。
示例代碼:
#include <stdio.h> #include <stdlib.h> int main() { int *p = NULL; p = (int *) malloc(sizeof(int)); *p = 3; printf("p=%p\n", p); printf("*p=%d\n", *p); p = (int *) realloc(p, sizeof(int)); printf("p=%p\n", p); printf("*p=%d\n", *p); p = (int *) realloc(p, 3 * sizeof(int)); printf("p=%p\n", p); printf("*p=%d", *p); // 釋放p指向的空間 realloc(p, 0); p = NULL; return 0; }
下面要注意的幾點是:
realloc()有可能操做失敗,返回NULL,因此不要把它的返回值直接賦值給原來的指針變量,以避免原值丟失:
#include <stdio.h> #include <stdlib.h> int main() { char *str = NULL; str = (char *)malloc(sizeof(char)); *str = 'a'; char *p = (char *)realloc(str, sizeof(char) * 10); if (p != NULL) { str = p; } printf("%s\n", str); return 0; }
參考: