Action()
{
// 脚本实现读取D盘下test.txt文件中内容并打印
char ch;
long file_stream;
char * filename = "D:\\test.txt";
// 判断是否可以读取D:\test.txt文件
file_stream = fopen(filename, "r");
if(file_stream == NULL)
{
//不能读取文件,输出错误信息
lr_output_message("不能打开%s文件",filename);
return -1;
}
//判断没有读取到文件结束符,执行循环体中内容
while(!feof(file_stream))
{
ch=fgetc(file_stream);
lr_output_message("%c",ch);
}
return 0;
}
问题:最后一个字符是k,实际输入多了一个奇怪字符,如何准确输入内容不包含奇怪字符??
解答思路:
修改代码: lr_output_message("%c",ch); -----> lr_output_message("%d",ch); 输出每个字符的值,得到:
eof并不是真正的结束,它需要遇到EOF时才会变成正值,此刻才是结束,也就是说,feof需要在读出EOF之后,才知道文件结束了,现在我们把feof的返回值输出来看一下它是在什么时候变值的。
通过对比,我们看到,在读取最后一个字符k后,feof()的返回值是0,此时while()条件成立,然后继续向文件后读,读出文件结束标志EOF,此刻feof()的返回值变位16,while()条件不满足,因此不再继续,但是由于读出了最后一个EOF,按照代码的要求,读一个马上输出来,因此要输出来,就出现了上文的多出来一个字符情况。
如果将上述代码中 ch=fgetc(file_stream); ---> fscanf(file_stream,"%c",&ch); ,即使用fscanf函数,出现如下结果:
和上面的分析情况相似,至于为什么最后一个字符输出的是k,而不是上面的EOF ,这就是fscanf的使用问题了,关于fscanf在遇到EOF时,应经不能读入有效字符了,但是此时输出流中的字符还是k,因此printf中又输出一遍k。
上述的问题就是常见的多输出问题,那么该怎么样解决这些问题呢,我在此提出两种解决办法
方法一:
这种解决办法只是写法上的巧妙变化,在前面的写法中,我们倾向于现在读入,现在输出,在这里我们提出一种新的思想,先读入,然后判断此时的指针位置是否合法,在合法的情况下输出上一次读入的值,然后在读取下一个,也就是先读然后判断最后输出的模式,我们每次输出的是上次的字符,因此在最后一个不合法的位置,我们输出了最后一个字符,之后就不会继续循环了。
方法二:
while(fscanf(file_stream,"%c",&ch)==1)
这种解决办法采用的是利用fscanf的返回值来判断文件是否结束,关于fscanf的返回值是:fscanf返回的是实际读取的数据个数,出错或者到结尾返回EOF。在读取最后一个字符时,fscanf不能读到有效字符,因此结果将会返回EOF,自然不会进入whie循环中,整体思路是先判断读到的字符是否合法,然后才输出 。两个方法的思路都是先判断读取的字符是否合法,合法再输出,这样就不会出现奇怪的字符啦。