getchar与setbuf之经典错误

        考虑下面的例子:

#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));


 

          

 


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
getchar和scanf都是C语言中用于从标准输入中获取字符的函数。 使用scanf函数读取字符时,可以使用格式化字符串"%c"来指定输入的字符类型。例如,scanf("%c", &a)将会读取一个字符并将其存储到变量a中。需要注意的是,scanf在读取字符后会将换行符留在输入缓冲区中,这可能导致后续的输入操作出现问题。 而getchar函数则直接从标准输入中读取一个字符,并返回读取到的字符。它不需要格式化字符串,也不会将换行符留在输入缓冲区中。因此,getchar函数更适合用于读取单个字符。 在使用getchar函数时,我们通常需要使用一个额外的getchar函数来读取输入缓冲区中的换行符。这是因为在前一个输入操作之后,输入缓冲区中可能仍然存在一个换行符。如果不读取它,那么后续的getchar函数可能会直接读取到这个换行符,而不是我们期望的字符。 总结起来,scanf函数可以读取指定类型的字符,并且需要注意处理输入缓冲区中的换行符。而getchar函数则更简单直接,适用于读取单个字符。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C语言 字符输入 scanf和getchar的区别](https://blog.csdn.net/weixin_73832459/article/details/127747670)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值