"cin"错误地解析了Mac上的"double"输入



一点背景 - 我是学习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::sortstd::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。

最新更新