首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C语言 >

关于C语言中feof函数的疑义,读到最后了为什么还读一遍

2012-08-03 
关于C语言中feof函数的疑问,读到最后了为什么还读一遍?feof不就是判断文件指针是否到最后了吗,结果多读了

关于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;
 //...
}
类似的例子还可以举很多。

热点排行