关于C语言中feof函数的疑问,读到最后了为什么还读一遍?
feof不就是判断文件指针是否到最后了吗,结果多读了一遍。
C语言代码:
- C/C++ code
#include <stdio.h>#include <stdlib.h>#include <string.h>struct stu{ int id; char name[20];};int main(){ struct stu stu1 = {1, "zhangsan"}; struct stu stu2 = {2, "lisi"}; FILE* fp = fopen("test.db", "ab+"); fwrite(&stu1, sizeof(struct stu), 1, fp); fwrite(&stu2, sizeof(struct stu), 1, fp); fclose(fp); fp = fopen("test.db", "rb"); printf("id\tname\n"); while(!feof(fp)) { struct stu s; fread(&s, sizeof(struct stu), 1, fp); printf("%d\t%s\n", s.id, s.name); }}结果:
idname
1zhangsan
2lisi
2lisi
就是说,最后一条写入文件的记录在读的时候多读了一遍。请问这是什么问题?
[解决办法]
可以判断一下 fread 的返回值。
[解决办法]
feof()用来侦测是否读取到了文件尾. 当fread第一次读取lisi时,还不知道是不是到了文件尾。所以循环还没结束。当fread再次读取时,发现到了文件尾,但还要执行printf(就是再次输出lisi)。
所以有了两次lisi。其实,直接用fread做循环判断条件即可。
[解决办法]
- C/C++ code
#include <stdio.h>#include <stdlib.h>#include <string.h>struct stu{ int id; char name[20];};int main(){ struct stu stu1 = {1, "zhangsan"}; struct stu stu2 = {2, "lisi"}; FILE* fp = fopen("test.db", "ab+"); fwrite(&stu1, sizeof(struct stu), 1, fp); fwrite(&stu2, sizeof(struct stu), 1, fp); fclose(fp); fp = fopen("test.db", "rb"); struct stu s; printf("id\tname\n"); while(fread(&s, sizeof(struct stu), 1, fp) == 1) { printf("%d\t%s\n", s.id, s.name); }}
[解决办法]
不要使用
while (条件)
更不要使用
while (组合条件)
要使用
while (1) {
if (条件1) break;
//...
if (条件2) continue;
//...
if (条件3) return;
//...
}
因为前两种写法在语言表达意思的层面上有二义性,只有第三种才忠实反映了程序流的实际情况。
典型如:
下面两段的语义都是当文件未结束时读字符
whlie (!feof(f)) {
a=fgetc(f);
//...
b=fgetc(f);//可能此时已经feof了!
//...
}
而这样写就没有问题:
whlie (1) {
a=fgetc(f);
if (feof(f)) break;
//...
b=fgetc(f);
if (feof(f)) break;
//...
}
类似的例子还可以举很多。