`

c++ ifstream 读文件 最后一行读两次

 
阅读更多

http://blog.csdn.net/rebel_321/article/details/4927464

 

用ifstream的eof(),竟然读到文件最后了,判断eof还为false。网上查找资料后,终于解决这个问题。

参照文件:http://tuhao.blogbus.com/logs/21306687.html

 

在使用C/C++读文件的时候,一定都使用过eof()这个函数来判断文件是否为空或者是否读到文件结尾了,也会在使用这个函数的过程中遇到一些问题,如不能准确的判断是否为空或者是否到了文件尾,以至于有些人可能还会怀疑这个函数是不是本身在设计上就有问题。

先来看看如下这段代码:

#include <iostream>
#include <fstream>

using namespace std;
int 
main()
{
  char ch = 'x'
;
  ifstream fin("test.txt" /*, ios::binary*/
);
  if (fin.eof())
  {
    cout << "file is empty."<<endl;
    return 0;

  }

  while (!
fin.eof())
  {
    fin.get
(ch);
    cout <<
 ch;
  }
    
  system("pause"
);
  return 0;
}

编译并运行以上代码,

如果test.txt不存在,程序会形成死循环,fin.eof()永远返回false,
如果test.txt为空,程序打印出一个x字符,
当test.txt中存在一字符串“abcd”且没有换行时,程序打印出“abcdd”,
当存在以上字符串并且有一新的空行时,程序打印出“abcd”加上一空行。

这种现象可能让很多人很迷惑,程序运行的结果似乎很不稳定,时对时错。使用binary模式读时结果一样。
在这里,大家可能有一个误区,认为eof()返回true时是读到文件的最后一个字符,其实不然,eof()返回true时是读到文件结束符0xFF,而文件结束符是最后一个字符的下一个字符。如下图所示:


 

因此,当读到最后一个字符时,程序会多读一次(编译器会让指针停留在最后一个字符那里,然后重复读取一次,这也就是就上面最后一个字符会输出两次的原因。至于是不是所有的编译器都这样处理我就不太清楚了,我使用的VC6,VC8似乎都是这样的)

问题出来了,就要找出对应的解决之道,要解决以上的问题,只需要调整一下条件语句即可:
                          fin.peek() == EOF      fin.get(ch)                                 

再来看一下另外一种情况:

#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main()
{
    string str;
    ifstream fin("test.txt"/*, ios::binary*/);
    if (fin.peek() == EOF)
    {
        cout << "file is empty."<<endl;
        return 0;
    }

    while (!fin.eof())
    {
        fin >> str;
        cout << str;
    }    
    system("pause");
    return 0;
}

 

 

上述代码在VC8下编译运行,发现,当文件结尾没有空行时,结果正确,当结尾有空行时,最后一个字符串将被重复输出一次, 而VC6的情况则有所不同,没有重复输出,但输出了一个空行。

因此,为了保证在不同的编译器下得到一致的我们期望的结果,将条件语句做一下修改:
                                           fin >> str                                                      

综上所述,我们可以得到以下结论:
1. 判断文件是否为空时使用peek函数,若peek返回EOF则文件为空;
2. 读取文件过程中,读取非char型时,使用peek判断文件尾将不再适用,循环判断条件应改用>>操作符进行读取,若读入char型缓冲区,peek函数会表现得很好。

 

 

peek() —— 此函数将返回输入流文件的下一个字符,但它不移动内置指针

分享到:
评论

相关推荐

    使用ifstream和getline读取文件内容[c++]

    c++、getline、每次读取一行txt; //读取方式: 逐词读取, 词之间用空格区分; //读取方式: 逐行读取, 将行读入字符数组, 行之间用回车换行区分; //读取方式: 逐行读取, 将行读入字符串, 行之间用回车换行区分

    c++输入文件流ifstream用法详解_ims的博客-CSDN博客1

    c++输入文件流ifstream用法详解_ims的博客-CSDN博客c++输入文件流ifstream用法详解原创阅读数 43446收藏发布于2018-05-18

    ifstream和getline读取文件

    ifstream和getline读取文件

    C++文件操作 C++ 文件操作

    ifstream: 读操作(输入)的文件类(由istream引申而来) fstream: 可同时读写操作的文件类 (由iostream引申而来) 打开文件(Open a file) 对这些类的一个对象所做的第一个操作通常就是将它和一个真正的文件联系...

    c++对asc码文件的存取操作

    c++中,文件的ASC码形式存储,用fstream 的ifstream和ofstream定义infile和outfile对象,对文件进行操作。

    利用ifstream和ofstream写的文件分割和合并

    利用ifstream和ofstream写的文件分割和合并 用c++中的纯fstream写的文件分割和合并程序, 大家可以看看,加强一下c++中的文件操作

    ofstream ifstream 文件操作

    ofstream ifstream 文件操作 是VC中的文件读入读出的基本操作!

    c++ 文件操作(fstream)

    c++ 文件操作 fstream ifstream ofstream

    C++中读取控制台输出,并将文件指针FILE*转换为istream

    NULL 博文链接:https://cherishlc.iteye.com/blog/2400974

    C/C++ 获取文件大小

    C++获取文件大小C++获取文件大小利用ifstream利用C函数利用Windows _stat函数参考文章 C++获取文件大小 利用ifstream 以下操作需要包含头文件fstream 和 string size_t GetFileSize(const std::string& file_name){...

    用C++实现简单的文件IO操作

    回写看起来像另一次输出, 然后调用方法关闭。像这样: fout ; fout.close();  现在你用文本编辑器打开文件,内容看起来是这样:  Here is a number: 150 Now here is a string: John Doe  很简单吧! 现在继续...

    C++中进行txt文件读入和写入的方法示例

    vs里面直接输入数据,这一般常见于简单算法和数据的处理,但是一旦处理大数据的话,几百万,几千万,上亿个数据手打似乎不能轻易实现的,那么这篇文章我们来搞懂C++环境下如何进行io流读取txt文件,其实我们需要一个...

    C++文件的使用

    由 于对类ofstream, ifstream 和 fstream 的对象所进行的第一个操作通常都是打开文件,这些类都有一个构造函数可以直接调用open 函数,并拥有同样的参数。这样,我们就可以通过以下方式进行与上面同样的定义对象和...

    ofstream,ifstream,fstream读写文件

    在编程总经常会用到读写文件,基本都是使用ofstream,ifstream,fstream

    VC++读写文件和注册表的源程序

    C++对文件读写的支持,ofstream和ifstream的用法。 Win32 SDK对文件读写的支持,CreateFile函数、WriteFile函数、ReadFile函数的使用;MFC对文件读写的支持,  CFile类和CFileDialog的使用,文件过滤器的设置。...

    C++格式化读写文件操作

    格式化读取:使用输入流对象(例如cin或ifstream对象)和格式控制符,按照特定的格式从文件中读取数据。可以使用&gt;&gt;运算符将数据读取到变量中。 格式化写入:使用输出流对象(例如cout或ofstream对象)和格式控制符...

    C++:使用getline读取文本文件

    之前在使用C++中的getline读取文本文件时由于没有仔细看getline的定义,导致出了错:在读取文本文件时未读取到文件中的第一行。 错误的源代码如下: vectorreadfile&#40;string s1,vectorv1&#41; { ifstream infile...

    ifstream.pdf

    ifstream.pdf ifstream.pdf ifstream.pdf

    c++/MFC文件拷贝

    用c++和MFC各自实现的文件拷贝功能的代码

Global site tag (gtag.js) - Google Analytics