getchar与setbuf之经典错误

本文深入解析了C语言编程中缓冲区管理的常见错误,具体讨论了`setbuf`函数的不当使用导致的缓冲区未正确清空问题。通过实例展示了将缓冲数组定义为静态数组或动态分配缓冲区的方法来解决此问题,确保了程序的正常运行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        考虑下面的例子:

#include <stdio.h>
int main()
{
	char c;
	while ((c = getchar()) != EOF){
		putchar(c)
	}
} 

        getchar函数在一般情况下返回的是标准输入文件中的下一个字符,当没有输入时返回EOF。这个程序乍一看似乎是把标准输入复制到标准输出,实则不然。原因在于程序中的变量c被声明为char类型,而不是int类型。这意味着c无法容下所有可能的字符,特别是,可能不发容下EOF。因此,最终结果存在两种可能:第一种可能是某些合法的输入字符在被“截断”后市的c的取值与EOF相同;另一种可能则是,c根本不可能娶到EOF这个值。对于前一种情况,程序将在文件复制的中途终止,对于后一种情况,程序将陷入死循环。

        解决办法:只需要把c声明为int即可。

        再看一个关于setbuf的例子:

        学过c语言的朋友都知道程序员可以自己控制缓冲区是否开启,这种控制能力是通过调用库函数setbuf/setvbuf实现的。如果buf是一个大小合适的字符数组,那么:              

setbuf(stdout, buf);

语句将通知输入/输出库,所有写入到stdout的输出都应该使用buf作为输出缓冲区,知道buf缓冲区被写满或者程序员直接调用fflush函数刷新缓冲区,buf缓冲区的内容才实际写到stdout中。缓冲区的大小由系统头文件<stdio.h>中的BUFSIZE定义。那么,请看例子:

#include <stdio.h>
int main()
{
	int c;
	char buf[BUFSIZE];
	setbuf(stdout, buf);
	while ((c = getchar()) != EOF){
		putch(c);
	}
}

        你找出这个错误了吗?很细微是吧?程序中对库函数setbuf的调用通知了输入/输出库所有字符的标准输出应该首先缓存在buf中。要找到这个错误出自哪里,我们不妨思考一下buf缓冲区最后一次被清空是在什么时候?答案是在main函数结束之后,作为程序交回控制给操作系统之前,C运行时库必须进行的清理工作的一部分。但是,在此之前buf字符数组已经被释放了!

        第一种解决办法:将缓冲数组定义为静态数组: static char  buf[BUFSIZE];

        第二种解决办法:动态分配缓冲区,在程序中并不主动释放分配的缓冲区。由于缓冲区是动态分配的,所以main函数结束时,并不会释放该缓冲区,这样C运行时库进行清理工作时就不会发生缓冲区已经释放的情况: 

char  *malloc();           
setbuf(stdout, malloc(BUFSIZE));


 

          

 


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值