C++ std::getline() 不设置 EOF 标志

C++ std::getline() doesn't set the EOF flag

假设我们有一个格式如下的文件A:

0 5
1 10
2 10
5 0
10 5

表示某个图的所有有向边,例如 0 > 5,依此类推。文件根据左节点排序。现在假设我们有一些机器,每台机器都在对一组节点进行一些计算,例如:

Machine 1: for {0,1}
Machine 2: for {2,5}
Machine 3: for {10}

如上所示,集合已排序,即机器 i 的节点索引小于机器 i+1。计算取决于从每个节点离开的边,所以我尝试逐行读取机器 1 的文件 A,然后是机器 2,然后是机器 3,当我到达离开的最后一个边缘时,它们中的每一个都退出循环机器负责的最后一个节点。该文件由分配作业的某些主机读取。对于上面的例子:

For Machine 1
Loop through lines 
0 5
1 10
Exit loop

For Machine 2
Loop through lines 
2 10
5 0
Exit loop

For Machine 3
Loop through lines 
10 5
Exit loop

我有以下 C++ 代码:

ifstream graph_file;
graph_file.open("graph.txt");

//...

string line;
unsigned long left_node;
while(1){

    //EOF reached. Break.
    if (graph_file.eof()){
        break;
    }

    getline(graph_file, line);
    left_node = stol(line);

}

假设我们已经完成了机器 1 和 2(代码未显示但工作正常)并且文件指针指向机器 3 应该开始的输入文件行。所以 std::getline() 读取上面示例的最后一行,并且在下一次迭代中循环不会中断,因为标志不是 EOF 所以 std::stol() 抛出 std::invalid_argument 因为它什么也没读到。

为什么会这样?

why eof() in a loop condition does not work 相同的原因。请注意,您的代码基本上与 while(!graph_file.eof()) 完全相同,因为您是在循环开始时检查条件,这正是普通 while 循环所做的。

eof() 仅在第一次尝试 读取文件末尾 时设置。如果末尾有换行符停止输入,则不会设置标志。

修复在这个答案顶部的 link 中。