目录
1.malloc函数
原型:
void *malloc(unsigned int num_bytes)。
头文件:
#include<stdlib.h>。
返回值:
如果分配成功则返回指向被分配内存的指针(无类型),否则返回空指针NULL。
注意事项:
(1)malloc函数的返回的是无类型指针,在使用时一定要强制转换为所需要的类型。
(2)在使用malloc开辟空间时,使用完成一定要释放空间,如果不释放会造内存泄漏。
(3)在使用malloc函数开辟的空间中,不要进行指针的移动,因为一旦移动之后可能出现申请的空间和释放空间大小的不匹配
2.calloc函数
原型:
void *calloc(size_t n, size_t size)
头文件:
同malloc,#include<stdlib.h>。
返回值:
同malloc,如果分配成功则返回指向被分配内存的指针(无类型),否则返回空指针NULL。。
注意事项:
大致用法和malloc基本相同,区别在于:
(1)比malloc函数多一个参数,n个size个字节的空间。
(2)calloc在申请后对空间逐一进行初始化,并设置值为0。
malloc/calloc使用示例:
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char* argv[], char* envp[])
{
int n;
int *numbers1;
if((numbers2=(int *)malloc(5*sizeof(int)))==NULL)//为numbers1在堆中分配内存空间
{
printf("malloc memory unsuccessful");
exit(1);
}//(若申请失败则异常退出)
printf("numbers1 addr: %8X\n",(int)numbers1);
free(numbers1);//释放numbers1
numbers1=NULL;//指向空指针
return 0;
}
注:calloc函数由于给每一个空间都要初始化值,那必然效率较malloc要低,并且现实世界,很多情况的空间申请是不需要初始值的。
3.realloc函数
原型:
void realloc(void *ptr, size_t new_size)
头文件:
同上#include<stdlib.h>。
返回值:
如果当前内存段后有足够的空间,realloc()返回原来的指针。
如果当前内存段后没有足够的空间,realloc()返回一个新的内存段的指针。
注意事项:
如果是将分配的内存扩大,则有以下情况:
(1)如果当前内存段后面有需要的内存空间,则直接扩展这段内存空间,realloc()将返回原指针。
(2)如果当前内存段后面的空闲字节不够,那么就使用堆中的第一个能够满足这一要求的内存块,将目前的数据复制到新的位置,并将原来的数据块释放掉,返回新的内存块位置。
(3)如果申请失败,将返回NULL,此时,原来的指针仍然有效。
(4)如果调用成功,不管当前内存段后面的空闲空间是否满足要求,都会释放掉原来的指针,重新返回一个指针,虽然返回的指针有可能和原来的指针一样,即不能再次释放掉原来的指针。
(5)如果将分配的内存减少,realloc仅仅是改变索引的信息。
realloc使用示例:
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char* argv[], char* envp[])
{
int input;
int n;
int *numbers1;
int *numbers2;
numbers1=NULL;
if((numbers2=(int *)malloc(5*sizeof(int)))==NULL)//为numbers2在堆中分配内存空间
{
printf("malloc memory unsuccessful");
exit(1);//若申请失败则异常退出
}
printf("numbers2 addr: %8X\n",(int)numbers2);
for(n=0;n<5;n++) //初始化
{
*(numbers2+n)=n;
//printf("numbers2's data: %d\n",*(numbers2+n));
}
printf("Enter new size: ");
scanf("%d",&input);
//重新分配内存空间,如果分配成功的话,就释放numbers2指针,
//但是并没有将numbers2指针赋为NULL,也就是说释放掉的是系统分配的堆空间,
//和该指针没有直接的关系,现在仍然可以用numbers2来访问这部分堆空间,但是
//现在的堆空间已经不属于该进程的了。
numbers1=(int *)realloc(numbers2,(input+5)*sizeof(int));
if(numbers1==NULL)
{
printf("Error (re)allocating memory");
exit(1);
}//分配失败则异常退出
printf("numbers1 addr: %8X\n",(int)numbers1);
/*for(n=0;n<5;n++) //输出从numbers2拷贝来的数据
{
printf("the numbers1's data copy from numbers2: %d\n",*(numbers1+n));
}*/
for(n=0;n<input;n++)//新数据初始化
{
*(numbers1+5+n)=n+5;
//printf("numbers1' new data: %d\n",*(numbers1+5+n));
}
printf("\n");
free(numbers1);//释放numbers1,此处不需要释放numbers2,因为在realloc()时已经释放
numbers1=NULL;
//free(numbers2);//不能再次释放
return 0;
}
示例中exit的用法:
C语言中exit(0)与exit(1)有什么区别_yyfwd的博客-CSDN博客_exit
(1)_exit()退出进程会清理I/O缓冲区,exit()结束进程进入到内核中。
exit(1)表示异常退出,这个1是返回给操作系统的不过在DOS好像不需要这个返回值 。
exit(0)表示正常退出 。
(2)()中的值返回给操作系统的,0是正常退出,其他值是异常退出,在退出前可以给出一些提示信息,或在调试程序中察看出错原因。
(3)return是语言级别的,它表示了调用堆栈的返回;而exit是系统调用级别的,它表示了一个进程的结束。
return是返回函数调用,如果返回的是main函数,则为退出程序。
exit是在调用处强行退出程序,运行一次程序就结束。
return是返回 , 函数返回而exit是退出。
参考博客: