从文件读取后输出中有额外的零行

Extra lines of zeros in output after reading from a file

我是一名正在尝试学习 C++ 的初级程序员。我正在尝试从文件中读取以下信息:

迪恩·德菲诺 88 98 99
莎莉·约翰逊 78 89 82
比尔·本尼 75 79 81
托马斯·比罗斯 78 84 89

但是,我得到以下输出,但我不明白为什么:

迪恩·德菲诺 88 98 99
0 0 0
莎莉·约翰逊 78 89 82
0 0 0

代码如下:

#include <fstream>
#include <iostream>
#include <iomanip>
using namespace std;

const int NAMESIZE = 15;
const int MAXRECORDS = 50;
struct Grades                             // declares a structure
{
    char name[NAMESIZE + 1];
    int test1;
    int test2;
    int final;

};

typedef Grades gradeType[MAXRECORDS];    
 // This makes gradeType a data type
 // that holds MAXRECORDS
 // Grades structures.

void readIt(ifstream&, gradeType, const int);

int main()

{    
     ifstream indata;
     indata.open("graderoll.dat");
     int numRecord = 4;                
     gradeType studentRecord; 

     if(!indata)
     {
        cout << "Error opening file. \n";
        cout << "It may not exist where indicated" << endl;
        return 1;
     }

    readIt(indata, studentRecord, MAXRECORDS); 

    // output the information 
        for (int count = 0; count < numRecord; count++)
    {
           cout << studentRecord[count].name << setw(10) 
            << studentRecord[count].test1
            << setw(10) << studentRecord[count].test2;
           cout << setw(10) << studentRecord[count].final << endl;
    }                

    return 0;
}

void readIt(ifstream& inData, gradeType gradeRec, const int max)

{
   int total = 0;

   inData.get(gradeRec[total].name, NAMESIZE);
   while (inData)
   {
     inData >> gradeRec[total].test1;
     inData >> gradeRec[total].test2;
     inData >> gradeRec[total].final;

     total++;    

     inData.clear(); 
     inData.get(gradeRec[total].name, NAMESIZE);


  }

}

有什么建议可以帮助我解决这个问题吗?

问题是您使用了:

while (inData)

将 iostream 转换为布尔值时,布尔值反映了 EOF 标志的状态。但是,直到 您尝试从文件末尾读取数据后,才会设置 EOF 标志。

那么发生的事情是您的代码成功读取文件的最后一行,然后循环并且 EOF 标志尚未设置,并尝试从中读取 more 行文件。这不会读取任何实际数据,您会得到零。之后,设置 EOF 标志并终止循环。

解决此问题的一种方法可能是在 读取预期数据后测试 EOF 标志,因此:

   while (true)
   {
     inData >> gradeRec[total].test1;
     inData >> gradeRec[total].test2;
     inData >> gradeRec[total].final;

     if (!inData) {
       break;
     }

     total++;    

     inData.clear(); 
     inData.get(gradeRec[total].name, NAMESIZE);
   }

您还可以稍微重新安排您的循环,这样您就没有代码来读取名称两次,一次在循环外,一次在循环结束处。

我解决了这个问题:

void readIt(ifstream& inData, gradeType gradeRec, const int max)

{
   int total = 0;

   inData.get(gradeRec[total].name, NAMESIZE);
   while (inData)
   {
     inData >> gradeRec[total].test1;
     inData >> gradeRec[total].test2;
     inData >> gradeRec[total].final;

     total++;    

     inData.ignore(NAMESIZE, '\n');
     inData.get(gradeRec[total].name, NAMESIZE);


  }

}

inData.ignore(NAMESIZE, '\n'); 将使字符数组正常工作。这是因为字符数组限制为 15 个字符。因此,如果指定忽略接下来的 15 个字符和/或直到遇到换行符转义序列,文件的每一行都会被正确读取。