feof()函数问题
今天写聊天室的时候,在读不定大小的文件的时候,想到了使用feof这个函数,但是在写测试程序的时候出了点问题,于是让我产生了对研究一下这个函数的强烈兴趣。那么就来简单的谈一下:
首先man一下feof给出:
The function feof() tests the end-of-file indicator for the stream pointed to by stream, returning nonzero if it is set.
翻译过来大概就是feof会检测文件末尾指示符是不是被置位,如果被置位,那么返回非0值,否则返回0值。
那么有下面几个问题,第一看代码:
main(){ char str[50]; FILE * fp = fopen("file","r"); while(1) { printf("the file point before fgets is %d\n",ftell(fp)); fgets(str,100,fp); printf("the file point after fgets is %d\n",ftell(fp)); printf("%s",str); if(feof(fp)) break; } fclose(fp); }
然后我们置file里的内容是这样的:
first second最后跑出来的结果如下:
the file point before fgets is 0 the file point after fgets is 6 first the file point before fgets is 6 the file point after fgets is 13 second the file point before fgets is 13 the file point after fgets is 13 second
是不是感觉很奇怪为什么second输出了两遍。那么再来看一个例子:对于一个空文件,如果我们将其打开以后,能否用feof判断文件是否结束?答案是不能,如果调用feof的话,会返回0值。也就是文件没有结束。
为什么会导致上面的一系列的事情呢,这就跟一开始说的那个文件末尾指示符有关,那么什么时候文件末尾的指示符会被置位呢。经过尝试得出的结论是,当文件指针第一次读到结尾以后,再进行读操作,才会进行置位操作。所以之前输出两次second的原因就是s指向的缓存区被printf了两遍的原因。
解决这个问题的方法比较简单,那就是将缓存区抹空,直接memset置为0即可,这样即使最后一次读也不会输出上次的缓存区里的内容。
简单谈谈如何实现判断文件是否为空,总结有以下几种方法:
1.通过fseek(fd, 0, SEEK_END)将文件指针放置在末尾,然后进行ftell(fd)函数,返回当前位置,如果为0就说明是空文件
2.通过stat一族的函数,找到文件结构体中的st_size元素,判断是否为0.
在用feof的时候是需要特别注意的。