一点背景 - 我是学习C++的新手。我正在通过加速C++并跟踪事情。在第 4 章中,我得到的代码并不像书中建议的那样完全有效,我已经仔细检查了我的所有代码,以确保我没有错误地键入他们示例中的内容(无论如何,我看不到任何错误)。
我有以下结构和两个函数:
struct Student {
std::string name;
double midtermGrade, finalGrade;
std::vector<double>homework;
};
std::istream& read(std::istream& is, Student& s) {
is >> s.name >> s.midtermGrade >> s.finalGrade;
std::cout << std::endl << "Name: " << s.name << std::endl
<< " - Md: " << s.midtermGrade << std::endl
<< " - Fn: " << s.finalGrade << std::endl;
read_hw(is, s.homework);
return is;
}
std::istream& read_hw(std::istream& in, std::vector<double>& hw) {
if (in) {
hw.clear();
double h;
while (in >> h) {
std::cout << " - Hw: " << h << std::endl;
hw.push_back(h);
}
in.clear();
}
return in;
}
在int main()
中,我像这样调用该代码:
Student record;
while (read(std::cin, record)) {
如果我编译并运行,然后粘贴(或键入)以下输入,我会得到以下输出:
// Input
Johnny 70 80 50 Fred 95 90 100 Joe 40 40 50
// Output
Name: Johnny
- Md: 70
- Fn: 80
- Hw: 50
Name: red
- Md: 95
- Fn: 90
- Hw: 100
Name: Joe
- Md: 40
- Fn: 40
- Hw: 50
如您所见,"弗雷德"变成了"红色"。我已经用一大堆名字尝试过这个;任何以 A-F 开头的内容都受到相同问题的影响,这让我怀疑这可能是十六进制问题。然而,"Nick"变成了"k","Xander"只是变成了"r"。
我完全不知所措。你知道发生了什么吗?
根据要求,这里有一个MCVE:
#include <iomanip>
#include <iostream>
#include <string>
#include <vector>
struct Student {
std::string name;
double midtermGrade, finalGrade;
std::vector<double>homework;
};
std::istream& read_hw(std::istream& in, std::vector<double>& hw) {
if (in) {
hw.clear();
double h;
while (in >> h) {
std::cout << " - Hw: " << h << std::endl;
hw.push_back(h);
}
in.clear();
}
return in;
}
std::istream& read(std::istream& is, Student& s) {
is >> s.name >> s.midtermGrade >> s.finalGrade;
std::cout << std::endl << "Name: " << s.name << std::endl
<< " - Md: " << s.midtermGrade << std::endl
<< " - Fn: " << s.finalGrade << std::endl;
read_hw(is, s.homework);
return is;
}
int main()
{
Student record;
std::vector<Student> students;
while (read(std::cin, record)) {}
return 0;
}
进一步研究后,似乎libc++
并没有像你一样从cin
中读取双精度。我不确定它是否只是专门针对双打,但我认为可能是。它似乎只影响OSX上的Clang编译器,这真的很烦人:P
对于TL;DR版本请看大卫的回答,我已经接受了。
我目前不知道有什么好的修复方法,但是如果您添加编译器标志-stdlib=libstdc++
则可以强制它使用较旧的,已弃用的stdlib,该stdlib将按预期工作。显然,它的功能较少(在我的示例中,缺少std::sort
和std::domain_error
)。其他选项包括不使用double
和对输入进行进一步解析(尽管因为我是C++新手,恐怕我没有那个史酷比)。
相关阅读:
- 尝试使用 C++ 、Ubuntu vs Mac 从 ASCII 文件中读取行...?
- C++ - 从cin 中读一个替身,后跟一个字符
- std::istringstream>> to 双重怪异行为
- http://clang-developers.42468.n3.nabble.com/g-4-2-8-vs-libc-bug-td4051616.html
- https://hg.mozilla.org/mozilla-central/raw-rev/edd76ca3717f
官方错误报告是在 3.5 年前打开的,至今仍未解决。
如果您对如何决定某物是否是double
有特定的要求,则必须编写实现这些要求的代码。该标准允许丢弃任何可能属于有效浮点表示形式的前导字符。像"e"、"i"、"n"和"f"这样的字母可以是有效浮点表示的一部分。
关于为什么这种编码从来都不是一个好主意,请参阅Falsehoods Programmers Believe About Names,特别是数字15。