考虑下面的例子:
#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));