
踩坑一:feof函数
函数原型:int feof( FILE *stream );
函数功能:如果调用操作尝试加载达到文件末尾的位置fscanf 返回值,feof函数返回非0,否则返回0(函数 feof 只用于测试流文件)
微软官网文档中关于feof描述:

部分翻译:当达到文件末尾时,读取操作返回文件结束指示符,直到流关闭或读取rewind,fsetpos,fseek或clearerr为止。例如,如果文件包括10个字节,你从文件中调用了10个字节,feof会返回0,
因为毕竟文件指针在文件的末尾,但是你没有尝试加载达到文件的末尾,只有在你尝试加载第11个字节以后feof函数才会返回非零值
换个表述就是:当文件外部位置指针对准文件末尾时,并未立即置位 FILE 结构中的文件结束标记,只有再执行一次读文件操作,才会置位结束标志,此后调用 feof 才会返回为真请读者注意下这句话:“只有再执行一次读文件操作,才会置位结束标志”
下面用断点调试来检测feof的返回值:
测试代码:


测试文本文件:

测试结果和运行结果:


从后面的测试结果可以看出,文本文件中唯有三个数字,但循环却执行了四次,在第三次循环时读取了第三个数字后,文件外部的位置指针指向了文件末尾,但是这时用feof函数判断的结果依然为0,并不会跳出循环,
只有当第四次循环中再用fscanf_s写入一次文件内容以后(此时fscanf_s的返回值是-1,也就是说读取是成功的,这也就是为什么第四次循环的ch的值没有变化),再用feof判断就会返回EOF(-1)
按照下面这些“先判定、再读取”写法,如果一个文件带有n个字节,那么while循环的外部操作会运行n+1次,如果不想多循环一次这么需要在while循环内部降低判断词语以及改成“先读取、再判定”
改法一:(while内部降低判断)

改法二:(while内部降低判断)


改法三:(先读取再判定)

踩坑二:ftell函数
函数原型:long ftell(FILE *fp);
函数功能:若变量调用失败,则返回文件位置指针当前位置相对于文件首的偏差字节数,否则返回-1L,(对于文本文件来说ftell的返回值是当前位置指针相对于文件起始位置的字节偏移量)
问题一:如果以追加读写的方法开启一个文本文件(文件内容为123),不进行任何I/O操作然后用ftell取得当前文件指针位置,请问ftell返回值为多少?
在微软的官网文档中有这么一段话:

红框中词语的含义:(当以追加的形式开启文件,在出现任何读取操作前文件指针移动到文件的末尾),如果以追加的方法开启文件且没有发生任何I/O操作,则文件指针在文件的结尾,鉴于本人英文水平有限没怎样读懂这句话
,感觉有点矛盾,下面就用断点调试来解答难题一

文本文件:

打开方式:


调试结果:

从后面的调试结果可以看出,当以追加读写的形式开启文件时,在未出现任何I/O操作时ftell的返回值为0,意思是文件指针在文件结尾,这个在文件开头的指针是下一次读取的位置,而不是下一次写入的位置(写入的位置在文件的末尾),意思是即使是用fscanf_s写入字节,那么调用的数组就是第一个字符fscanf 返回值,如果是读取数组那么就是在文件的最后一个字符的后一位写入(就是文件的末尾),下面放一张图方便理解

再回到前面那句话,(当以追加的形式开启文件,在出现任何读取操作前文件指针移动到文件的末尾),如果以追加的方法开启文件且没有发生任何I/O操作,则文件指针在文件的结尾,这句话的前半句说的在文件末尾的
文件指针指的是读取的指针,而在文件开头的指针是调用的指针(并不是说读取和写入用的两个不同的指针,是同一个指针,如果是两个不同的指针的话这么难以确认ftell是返回的读指针还是写指针的位置)

意思是即使以追加读写的方法开启文件
,如果出现读操作这么读取的是第一个字符,如果出现写操作这么写入的位置是文件的末尾(也就是最后一个字符的后一位)
补充:如果以下面为前提发生了一次写入操作,此时ftell的返回值为多少?
测试文件:

测试代码:

断点调试结果:



从后面断点调试的结果可以看出,ftell在出现I/O操作前返回值是0,也就是下一次读取的位置,在出现一次写入操作后,ftell的返回值是4,也就是一下次写入的位置,如下图
本文来自电脑杂谈,转载请注明本文网址:
http://www.pc-fly.com/a/jisuanjixue/article-122890-1.html
好好伤心