先看下面的程序
#include<iostream>
#include<fstream>
using namespace std;
int main()
{
ofstream outfile("test.txt");
outfile<<"abcdefg";
outfile.close();
char c;
ifstream infile("test.txt");
while(!infile.eof())
{
infile>>c;
cout<<c;
}
system("pause");
return NULL;
}
其输出是" abcdefgg ",可以发现输出了两个“ g ”,这是什么原因呢?
1、文件指针
当打开一个文件时,对文件输入流来说,指针指向的是当前读取字符串的下一位,每读一位,指针就往下移一位。
2、关于EOF
大家通常误认为文件结束符EOF是从文件读取的一个字符,其实不然,EOF被定义为int型的一个负数(比如-1)。EOF也不是文件中实际存在的内容,而是文件流的状态标志。在C++中,当读取文件失败才产生EOF,所以当输出第一“ g ”时,产生EOF,输出第二个“ g ”时,读取到EOF,循环结束。
3、解决方案
将判断循环结束的方式改为“ infile.peek() != EOF ”。
infile.peek() 其返回值是一个char型的字符,其返回值是指针指向的当前字符,但它只是观测,指针仍停留在当前位置,并不后移。
修改程序如下: 其输出结果是”abcdefg“
#include<iostream>
#include<fstream>
using namespace std;
int main()
{
ofstream outfile("test.txt");
outfile<<"abcdefg";
outfile.close();
char c;
ifstream infile("test.txt");
while(infile.peek() != EOF)
{
infile>>c;
cout<<c;
}
system("pause");
return NULL;
}
注:程序修改之后,若输入字符串最后有一个空格符或换行符,则输出结果是"abcdefgg",因为输出"g"时,文件还没有结束(还有一个空格符或换行符),而peek()函数不会删除之前缓存区里的字符"g",所以循环继续,输出的还是"g"。