注意:请在回答之前阅读评论。问题似乎是编译器特定的。
我有一个简单的程序,它从文件或控制台读取名称和一些等级到结构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;
}
编辑:添加换行符。
答案 0 :(得分:8)
当涉及失败的浮点格式化输入时,它们如何实现标准的实现之间存在不一致。
clang
(或者更确切地说,libc++
)读取并丢弃有效浮点表示可能包含的所有字符,即使它不能在此位置包含它们,并且转换必然会失败。这些字符包括C
和A
(大写和小写,因为它们是十六进制数字,标准实际上允许使用十六进制浮点数表示法)。因此,当尝试读取double
并且输入包含Carl
时,将读取并丢弃字符C
和A
,即使没有任何浮点数也可以从这些人物。
另一方面,只要明确转换失败,gcc
(libstdc++
)就会停止阅读。因此,如果输入包含Carl
,则转换将在第一个字符处停止(并保留在流中),因为十六进制浮点数不能以C
开头(它必须以{{开头) 1}})。
我不会自愿提出关于哪种实施方式正确无误的意见。无论它是什么,正常的代码应该避开语言的微妙和神秘的角落。如果学生记录是一行,那么代码应该读取行。如果学生记录被定义为“一个姓名和一系列数字,直到下一个名字”,则停止并阅读this article。
答案 1 :(得分:1)
我认为问题在于使用end of line
和double
解析输入。
我找到了两种解决方法:
阅读作业直至行尾。
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.peek()!='\n') && input >> x)
{
//std::cout << x << "\n";
S.homework.push_back(x);
}
input.clear();
}
return input;
}
如果您知道输入是整数
,请不要将其解析为doublefriend std::istream &operator>>(std::istream &input, Student_info &S)
{ // overloads operator to read into Student_info object
input >> S.name >> S.midterm >> S.final;
int x;
if (input)
{
S.homework.clear();
while (input >> x)
{
//std::cout << x << "\n";
S.homework.push_back(x);
}
input.clear();
}
return input;
}
答案 2 :(得分:1)
当您知道Student_Info
对象的信息总是在一行中时,最好使用以下策略。
std::istringstream
。Student_Info
。std::istringstream
对应的数据
醇>
它只是使解析代码更简单,更不容易出错。
// overloads operator to read into Student_info object
friend std::istream &operator>>( std::istream &input, Student_info &S )
{
std::string line;
if ( !getline(input, line) )
{
// Problem reading a line of text.
return input;
}
std::istringstream str(line);
str >> S.name >> S.midterm >> S.final;
S.homework.clear();
double x;
while ( str >> x)
{
S.homework.push_back(x);
}
return input;
}
FWIW,我无法复制您所看到的问题。查看http://ideone.com/13wHLa处的工作版本。