C语言之free函数及野指针

【FROM MSDN && 百科】

原型:  void free(void *ptr);

#include<stdlib.h>或#include <malloc.h>

Deallocate space in memory

释放ptr指向的存储空间。被释放的空间通常被送入可用存储区池,以后可在调用malloc、realloc以及realloc函数来再分配。

注意:连续两次使用free函数,肯定会发生错误。malloc的次数要和free的次数相等。


A block of memory previously allocated using a call to malloccalloc or realloc is deallocated, making it available again for further allocations.

If ptr does not point to a block of memory allocated with the above functions, the behavior is undefined.
If ptr is a null pointer, the function does nothing

Notice that this function does not change the value of ptr itself, hence it still points to the same (now invalid) location


DEMO:

//#define  FIRST_DEMO
#define  SECOND_DEMO
#ifdef FIRST_DEMO
#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
int main(void)
{
	int *buffer1,*buffer2,*buffer3;
	buffer1=(int *)malloc(100*sizeof(int));
	buffer2=(int *)calloc(100,sizeof(int));
	buffer3=(int *)realloc(buffer2,500*sizeof(int));
	free(buffer1);
	free(buffer3);
	getch();
	return 0;
}
#elif defined SECOND_DEMO
#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
int main(void)
{
	char *str;
	/*allocate memory for string*/
	str=(char *)malloc(10);
	if (str==NULL)
	{
		perror("malloc"); 
		abort();
	}
	else
	{
		/*copy "hello" to string*/
		strcpy(str,"hello");
		/*display string*/
		printf("String is %s\n",str);
		/*free memory*/
		free(str);
	}
	getch();
	return 0;
}
#endif

DEMO:perror

perror( ) 用来将上一个函数发生错误的原因输出到标准设备(stderr)。参数 s 所指的字符串会先打印出,后面再加上错误原因字符串。此错误原因依照全局变量errno 的值来决定要输出的字符串。

#include <stdio.h>
#include <stdlib.h>   //perror包含在此文件中
#include <conio.h>
int main(void)
{
    FILE *fp;
	fp=fopen("abc","r+");
	if (NULL == fp)
	{
		perror("abc");
	}
	getch();
	return 0;
}

output:

abc: No such file or directory


DEMO:

#include <stdio.h>
#include <conio.h>
#include <stdlib.h>
#include <string.h>
int main(void)
{
	char *ptr;
	ptr=(char *)malloc(100);
	strcpy(ptr,"Hello");
	free(ptr);    // ptr 所指的内存被释放,但是ptr所指的地址仍然不变,原来的内存变为“垃圾”内存(不可用内存)
#if 1
	if (ptr!=NULL)    /* 没有起到防错作用*/
	{
		strcpy(ptr," world");
		printf("%s\n",ptr);
	}
#endif
	getch();
	return 0;
}

【FROM: http://blog.csdn.net/onestep365/article/details/1897626】

free(str)后指针仍然指向原来的堆地址,即你仍然可以继续使用,但很危险,因为操作系统已经认为这块内存可以使用,他会毫不考虑的将他分配给其他程序,于是你下次使用的时候可能就已经被别的程序改掉了,这种情况就叫“指针”,所以最好free()了以后再置空
str = NULL;
即本程序已经放弃再使用他。

何谓“指针”,在这里补充一下。

野指针是指程序员或操作者不能控制的指针。野指针不是NULL指针,而是指向“垃圾”的指针。

造成“野指针”的原因主要有

1.指针变量没有初始化任何指针变量刚被创建时不会自动成为NULL指针,它的缺省值是随机的,它会乱指一气。在初始化的时候要么指向合法的指针,要么指向NULL。

2.指针变量被free或delete之后,没有设置为NULL。它们只是把指针所指的内存给释放掉,但并没有把指针本身干掉。通常会用语句if (p != NULL)进行防错处理。很遗憾,此时if语句起不到防错作用,因为即便p不是NULL指针,它也不指向合法的内存块。上文DEMO则是这种情况。

3.指针操作超越了变量的作用范围。  注意其生命周期。


【下面是摘自论坛里面的形象比喻,加深理解。】
CRT的内存管理模块是一个管家。   
你的程序(简称“你”)是一个客人。   
管家有很对水桶,可以用来装水的。   
malloc的意思就是“管家,我要XX个水桶”。   
管家首先看一下有没有足够的水桶给你,如果没有,那么告诉你不行。如果够,那么登记这些水桶已经被使用了,然后告诉你“拿去用吧”。   
free的意思就是说:“管家我用完了,还你!”。   
至于你是不是先把水倒干净才给管家,那么是自己的事情了。--是不是清零。   

管家也不会将你归还的水桶倒倒干清(他有那么多水桶,每个归还都倒干净岂不累死了)。反正其他用的时候自己会处理的啦。   
free之后将指针清零只是提醒自己,这些水桶已经不是我的了,不要再完里面放水了,^_^   
如果free了之后还用那个指针的话,就有可能管家已经将这些水桶给了其他人装饮料的了,你却往里面撒了泡尿。好的管家可能会对你的行为表示强烈的不满, 杀了你(非法操作)--这是最好的结果,你知道自己错了(有错就改嘛)。一些不好的管家可能忙不过来,有时候抓到你作坏事就惩罚你,有时候却不知道去那里 了--这是你的恶梦,不知道什么时候、怎么回事情自己就死了。不管怎么样,这种情况下很有可能有人要喝尿--不知道是你的老板还是你的客户了.^_^。   
所以啊,好市民当然是还了给管家的东西就不要再占着啦,.^_^。

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值