feof的处理

feof()函数是我们在C语言中操作文件经常用到的一个函数。这个函数用来表示我们是否已经到了文件的末尾的下一个位置。不管是二进制文件,还是文本文件它都管用。对比EOF(一般宏定义为-1),EOF只能用来判断文本文件是否到达末尾,因为文本文件中的数字是用ASCII表示的,ASCII字符的取值范围是0~255。而二进制文件中可能存在-1,所以不能用EOF来判断结束。

我们使用feof经常遇到的一个问题是,用fgets读文件,然后用fputs打印。最后一行会打印两遍。

这是为什么了?

先说一下feof()函数

在stdio.h里有feof的定义:
#define _IOEOF 0x0010
#define feof(_stream) ((_stream)->_flag & _IOEOF)

由此可知只有当_flag=_IOEOF时,feof()才会返回1。在VC里,只有当 file position indicator(在Windows上是fp->_ptr)到了文件末尾,然后再发生读/写操作时,fp-> _flag才会被置为含有_IOEOF,然后再调用feof(),才会得到文件结束的信息。并不是file position indicator一指到文件尾,feof()就认为文件结束。也就是说只有文件指针到了文件末尾的下一个位置,feof()才会返回1。

程序如下:

 while(!feof(fp))
 { 

  fgets(str,80,fp);
  fputs(str,stdout);
 }

那么我们读最后一行之后,打印最后一行,这个时候feof()返回的是0.接着使用fgets读文件,由于fgets()函数在读文件时,会判断是否碰到了_IOEOF,如果一开始就碰到了,那么传递的数组的内容不会改变,也就是说还是最后一行的内容。这样最后一行就会打印两遍。fgets()函数定义在文章的最后面。

那么如何解决这个问题了。

只要正确使用feof()即可,如何正确使用呢?记住一个原则:先读取内容,然后用feof()判断。这样你在文件最后一行之后读文件,feof()会返回1。就不会打印了,修改之后的程序如下:

 fgets(str,80,fp);
 while(!feof(fp))
 {
  fputs(str,stdout);
  fgets(str,80,fp);
 }

 

解决方法二:不用feof(),使用fgets()的返回值判断,fgets如果读到了文件尾,则会返回NULL,说明碰到了文件结束。

  
 while(fgets(str,80,fp))

 {
  fputs(str,stdout);
 }

这样也能成功打印文件内容。

 

 

fgets.c文件里面的内容,这个应该是fgets函数的定义。

_TSCHAR * __cdecl _fgetts (
        _TSCHAR *string,
        int count,
        FILE *str
        )
{
        REG1 FILE *stream;
        REG2 _TSCHAR *pointer = string;
        _TSCHAR *retval = string;
        int ch;

        _VALIDATE_RETURN(( string != NULL ) || ( count == 0 ), EINVAL, NULL);
        _VALIDATE_RETURN(( count >= 0 ), EINVAL, NULL);
        _VALIDATE_RETURN(( str != NULL ), EINVAL, NULL);

                if (count == 0)
                {
                        return NULL;
                }

       

       
        stream = str;

        _lock_str(stream);
        __try {

#ifndef _UNICODE
        _VALIDATE_STREAM_ANSI_SETRET(stream, EINVAL, retval, NULL);
#endif 
        if(retval!=NULL)
                {
                        while (--count)
                        {
                                        if ((ch = _fgettc_nolock(stream)) == _TEOF)
                                        {
                                                        if (pointer == string) {
                                                                        retval=NULL;
                                                                        goto done;
                                                        }

                                                        break;
                                        }

                                        if ((*pointer++ = (_TSCHAR)ch) == _T('\n'))
                                                        break;
                        }
                *pointer = _T('\0');
                }



done:

        ; }
        __finally {
                _unlock_str(stream);
        }

        return(retval);
}

posted @ 2014-03-22 14:07  Sprink  阅读(665)  评论(0编辑  收藏  举报