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

转载自: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

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);
    }

 

 

 

 

 

转载于:https://www.cnblogs.com/dongzhuangdian/p/5493735.html

你可能感兴趣的:(文件结束的判断和结束符的理解)