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; }
参考: