注意:在回答之前,请先阅读评论。问题似乎是特定于编译器的。
我有一个简单的程序,该程序从文件或控制台中读取名称和某些成绩,并将其读入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++
)读取并丢弃有效浮点表示形式可能包含的所有字符,即使该字符不能在此位置包含它们并且转换必然会失败。这些字符包括C
和A
(大写和小写,因为它们是十六进制数字,并且标准实际上允许使用十六进制浮点数表示法)。因此,当尝试读取double
且输入包含Carl
时,将读取并丢弃字符C
和A
,即使没有浮点数可以以这两个字符中的任何一个开头。
另一方面,一旦清楚转换将失败,gcc
(libstdc++
)就会停止读取。因此,如果输入包含Carl
,则转换将在第一个字符处停止(并且保留在流中),因为十六进制浮点数不能以C
开头(必须以0X
开头)。
对于哪种实现形式正确,我不会提出任何意见。无论是哪种情况,普通代码都应避开该语言的微妙和神秘的角落。如果学生记录是一行,则代码应读取行。如果将学生记录定义为“一个名字和一个数字序列,一直持续到下一个名字”,则停止并阅读this article。