• 文件结束的判断和结束符的理解


    转载自:http://blog.csdn.net/zhy10/article/details/1562649

    转载自:http://blog.csdn.net/sambian/article/details/644360

    EOF,即end of file,文件结尾,作为文件结束的标志,在程序中常作为判断的一个标志。但在我们平常的程序中却常发生意想不到的结果。
    下面这段程序,猜猜它输出的是什么?
    char c;
    ifstream fin("d://dat");//设d:/dat文件已存在,内容为ab。
    while(!fin.eof())
    {
        fin >> c;
        cout << c;
    }
    输出结果是abb,没想到吗?你可能会问,再输出第一个b的时候,文件指针已经指向了EOF,为何不结束?
    问题的关键是文件EOF机制是怎样运作的。

    我们来谈三个问题:

    1、文件指针
    当打开一个文件时,文件指针位置为0,并不是指向第一个字符,即第一个字符的位置为1。这一点我们可以通过peek()函数验证。peek()返回的是当前文件指针下一个位置的字符。所以有:
    ofstream fo("d://dat");
    fo << 'h';
    fo.close();
    ifstream fi("d://dat");
    char temp = fi.peek();
    cout << temp;
    会显示h。
    还有,用fo.seekp(0,ios::beg),得文件指针为0;fo.seekp(0,ios::end),得文件指针指向最后一个字符。

    2、关于EOF
    很多朋友认为文件尾有EOF,这是错误的。EOF是流的状态标志。在 C++中,是在读取文件失败时才产生EOF。所以第一个程序中,在输出第一个b时,产生了EOF,再输出第二个b时读取到EOF,循环结束。

    3、解决EOF困惑的办法
    我感觉在判断文件结束上,最好的方法就是判断文件指针相对于开头的位置,是否等于文件长度。即:
    long filelen;
    ifstream fin("d://dat");//设d:/dat文件已存在,内容为ab。
    fin.seekg(0,ios::end);
    filelen = fin.tellg();//获取文件长度
    fin.seekg(0,ios::beg);
    while (1)
    {
        if (filelen == fin.tellg())//到达文件尾,即指向EOF
        {
            flag = true;
            break;
        }
        读取数据...
    }

    当然还有别的方法,就是用peek()的预读性。
    peek()返回当前文件指针下一个位置的字符,而指针位置不变。所以我们可以这样:
    while (fi.peel()!=EOF)
    {
        ...
    }
    当while循环体中,文件指针指向最后一个字符,若没有fi.peel()!=EOF,则需要再下一个循环中才能触发EOF。而加了fi.peel()!=EOF后,用预读的方法检测出了EOF。

    #include <stdio.h>

    int main()
    {
        FILE *in,*out;
        char ch,infile[10],outfile[10];

        printf("Enter the infile name:");
        scanf("%s",infile);
        in=fopen(infile,"r");
        
        if(in==NULL)
        { 
            printf("Can't open the file that you want read!");
            return 1;
        }

        printf("Enter the outfile name:");
        scanf("%s",outfile);
        out=fopen(outfile,"w");
        if(out==NULL)
        { 
            printf("Can't open the file that you want to write!");
            return 1;
        }

        while(!feof(in))
        { 
            ch=fgetc(in);
            putchar(ch);
            fputc(ch,out);
        }
     
        fclose(in);
        fclose(out);
        
        return 0;
    }

    上面这个小程序,每次运行后,目标文件会比源文件多一个字节,比如,源文件Test1.txt的内容是:
    hehe
    运行后,目标文件Test2.txt的内容却是:
    hehe
    用记事本打开看的,多了一个字节,变成了5字节

    上面这段程序在谭浩强的C程序设计(第二版)中也有这个问题,这实际上是对feof这个函数的处理方式不理解所造成的,实际上:

    当feof(FILE *)读到EOF标志并不认为文件结束了,依旧返回0,直到读到EOF的下一个字符才返回1,这时才认为是文件结束。

    因此若以while(!feof(fp))为循环条件的时候,要将一个文件(fp)完全复制到另一个文件(fp1),需要加上判断if(ch!=-1),如下:


    while(!feof(in))
     {
      ch=fgetc(in);
      if(ch!=-1)
      fputc(ch,out);

      }

    或者:

     while(true)
        { 
            ch=fgetc(in);
            if(feof(in))
             break;
            putchar(ch);
            fputc(ch,out);
        }

     
     
  • 相关阅读:
    算法导论第十八章 B树
    腾讯2016春招之算法编程解析
    LeetCode:5_Longest Palindromic Substring | 最长的回文子串 | Medium
    搜狗2016校园招聘之算法编程解析
    linux sed在某些字符串的下一行插入内容?sed在下一行插入?
    linux shell搜索某个字符串,然后在后面加上字符串?字符串后面插入字符串?sed字符串后面插入字符串?
    linux环境中,如何使用tar来创建压缩包?解压缩?
    linux环境中,ssh登录报错,Permission denied, please try again.
    linux环境中安装NRPE插件执行远程"本地资源"检查?NRPE安装?
    linux环境安装nagiosgraph将nagios的性能数据绘制成动态图表?
  • 原文地址:https://www.cnblogs.com/dongzhuangdian/p/5493735.html
Copyright © 2020-2023  润新知