注意:在回答之前,请先阅读评论。问题似乎是特定于编译器的。

我有一个简单的程序,该程序从文件或控制台中读取名称和某些成绩,并将其读入struct Student_info结构中,然后通过重载<>运算符来打印出一些数据。但是,该程序将切断部分甚至整个单词,并转移数据。例如,输入

Eunice 29 87 42 33 18 13
Mary 71 24 3 96 70 14
Carl 61 12 10 44 82 36
Debbie 25 42 53 63 34 95

退货
Eunice: 42 33 18 13
Mary: 3 96 70 14
rl: 10 44 82 36
25: 63 34 95

这表明该流忽略了Carl的前两个字母,然后将整个流左移了1个字。我一直在尝试调试一个小时的大部分时间,但这似乎是任意的。对于不同的名称,将截断不同的单词,没有明显的样式。
#include <iostream>
#include <fstream>
#include <string>
#include <vector>

struct Student_info {
    friend std::ostream &operator<<( std::ostream &output,
                                    const Student_info &S ) { // overloads operator to print name and grades
        output << S.name << ": ";
        for (auto it = S.homework.begin(); it != S.homework.end(); ++it)
            std::cout << *it << ' ';
        return output;
    }

    friend std::istream &operator>>( std::istream &input, Student_info &S ) { // overloads operator to read into Student_info object
        input >> S.name >> S.midterm >> S.final;
        double x;
        if (input) {
            S.homework.clear();
            while (input >> x) {
                S.homework.push_back(x);
            }
            input.clear();
        }
        return input;
    }

    std::string name; // student name
    double midterm, final; // student exam scores
    std::vector<double> homework; // student homework total score (mean or median)

};


int main() {
    //std::ifstream myfile ("/Users/.../Documents/C++/example_short.txt");
    Student_info student; // temp object for receiving data from istream
    std::vector<Student_info> student_list; // list of students and their test grades
    while (std::cin >> student) { // or myfile >> student
        student_list.push_back(student);
        student.homework.clear();
    }
    for (auto it = student_list.begin(); it != student_list.end(); ++it) {
        std::cout << *it << '\n';
    }
    return 0;
}

编辑:添加了换行符。

如您所见,它不适用于clang,但适用于GCC

最佳答案

涉及浮点格式失败的输入时,它们在实现标准方面的实现方式之间存在不一致之处。
clang(或更准确地说,libc++)读取并丢弃有效浮点表示形式可能包含的所有字符,即使该字符不能在此位置包含它们并且转换必然会失败。这些字符包括CA(大写和小写,因为它们是十六进制数字,并且标准实际上允许使用十六进制浮点数表示法)。因此,当尝试读取double且输入包含Carl时,将读取并丢弃字符CA,即使没有浮点数可以以这两个字符中的任何一个开头。

另一方面,一旦清楚转换将失败,gcc(libstdc++)就会停止读取。因此,如果输入包含Carl,则转换将在第一个字符处停止(并且保留在流中),因为十六进制浮点数不能以C开头(必须以0X开头)。

对于哪种实现形式正确,我不会提出任何意见。无论是哪种情况,普通代码都应避开该语言的微妙和神秘的角落。如果学生记录是一行,则代码应读取行。如果将学生记录定义为“一个名字和一个数字序列,一直持续到下一个名字”,则停止并阅读this article

09-10 00:19
查看更多