再读写文件时,我们除了会使用读写函数,文件打开方式之外判定一个文件读取结束也非常重要
文件读取结束的标志
1.文本文件结束的标志
文本文件读取是否结束,观察使用读文件函数的返回值;
使用fgetc时看返回值是否是EOF,EOF转换成数值就是-1,如果返回值是-1那么读取就结束了。为什么是EOF呢?这也很好理解,因为fgetc读取的是一个一个字符,而由ASCLL码表可以知道,字符可以转换成一些整型数值,那么正是因为EOF也对应着一个-1的整型数值,我们才可以把它作为读取结束的标志;
使用fgets的时候看返回值是否是NULL,我们知道,fgets是对文本行的读取,文本行说白了就是一行字符串,而字符串的操作我们一般是通过这个字符串首元素的地址来实行的,那么既然是地址,那就有着指针的意味,既然是指针,那用NULL这个空指针来判断是否读取结束不是正好吗?
2.二进制读取结束的标志
判断返回值是否小于实际上要读的个数
后续会一一例举;
3.被错误使用的feof
关于feof函数,很多使用者将这个函数的返回值作为判断文件是否结束的标志;但实际上这个函数的真正作用是在文件读取结束之后判断结束的原因是不是读到了文件末尾;也就是说这个函数的使用前提是文件读取已经结束了,它判断的是文件读取结束的原因;
文件读取结束的原因有两个:一是正常读取到文件末尾结束;二是在读取过程中读取出错异常终止;
而feof函数就是来判断文件的结束是不是正常的;对立地文件读取异常终止而结束使用的函数是ferror
代码示例:(feof和ferror函数也同时展示在里面)
文本文件读取结束:
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int c; // 注意:int,⾮char,要求处理EOF
FILE* fp = fopen("test.txt", "r");
if (!fp) {
perror("File opening failed");
return EXIT_FAILURE;
}
//fgetc 当读取失败的时候或者遇到⽂件结束的时候,都会返回EOF
while ((c = fgetc(fp)) != EOF) // 标准C I/O读取⽂件循环
{
putchar(c);
}
//判断是什么原因结束的
if (ferror(fp))
puts("I/O error when reading");
else if (feof(fp))
puts("End of file reached successfully");
fclose(fp);
}
二进制文件读取结束:
#include <stdio.h>
enum { SIZE = 5 };
int main(void)
{
double a[SIZE] = { 1.,2.,3.,4.,5. };
FILE* fp = fopen("test.bin", "wb"); // 必须⽤⼆进制模式
fwrite(a, sizeof * a, SIZE, fp); // 写 double 的数组
fclose(fp);
double b[SIZE];
fp = fopen("test.bin", "rb");
size_t ret_code = fread(b, sizeof * b, SIZE, fp); // 读 double 的数组
if (ret_code == SIZE) {
puts("Array read successfully, contents: ");
for (int n = 0; n < SIZE; ++n)
printf("%f ", b[n]);
putchar('\n');
}
else { // error handling
if (feof(fp))
printf("Error reading test.bin: unexpected end of file\n");
else if (ferror(fp)) {
perror("Error reading test.bin");
}
}
fclose(fp);
}