首先,您需要知道 C++ 中有两 getline
件事,一件在 I/O 区域,一件在顶级标准命名空间。
cin.getline(breed, 100)
是 I/O 区域中的一个(特别istream::getline()
是它对字符串一无所知,更喜欢处理字符数组。你应该避免那个。
如果您不想回到 C 遗留“字符串”的糟糕旧时代,那么了解字符串的那个通常是首选的那个。std::getline()
<<
此外,在 C++ 中混合使用特定类型的输入(如)和特定于行的输入(如getline
)操作时需要小心。了解每次操作前后文件指针的位置很重要。
例如,cin << someInt
将在它读入的整数之后立即离开文件指针。这意味着,如果您的下一个操作是getline()
,它可能会在该整数之后的行中找到所有内容(至少,这将是您的换行符输入以处理整数),而不是您将要输入字符串的下一行。
对于您的情况,一个简单的解决方法是在尝试获取下一行之前忽略包括换行在内的所有内容。这可以通过以下方式完成ignore()
:
#include <iostream>
#include <string>
#include <limits>
using namespace std;
int main() {
int birthyear; string breed; bool vaccines;
cout << "Please enter value for dog's birth year: ";
cin >> birthyear;
cout << "What is the breed of the dog: ";
cin.ignore(numeric_limits<std::streamsize>::max(), '\n');
getline(cin, breed);
cout << "Has the dog been vaccinated (1 = Yes/ 0 = No): ";
cin >> vaccines;
// Output what you got.
cout << birthyear << " '" << breed << "' " << vaccines << '\n';
}
您还可以选择确保所有输入都是基于行的(一旦输入这些行就将它们转换为正确的类型),因为这可能会减轻您确保指针位于正确位置以及输入错误(如输入xyzzy
整数)可以更好地处理。
这样的事情应该是一个好的开始:
#include <iostream>
#include <string>
#include <limits>
#include <set>
#include <cstdlib>
using namespace std;
// Get string, always valid. Optionally strip leading and
// trailing white-space.
bool getResp(const string &prompt, string &val, bool strip = false) {
cout << prompt;
getline(cin, val);
if (strip) {
val.erase(0, val.find_first_not_of(" \t"));
val.erase(val.find_last_not_of(" \t") + 1);
}
return true;
}
// Get unsigned, must ONLY have digits (other than
// leading or trailing space).
bool getResp(const string &prompt, unsigned long &val) {
string str;
if (! getResp(prompt, str, true)) return false;
for (const char &ch: str)
if (! isdigit(ch)) return false;
val = strtoul(str.c_str(), nullptr, 10);
return true;
}
// Get truth value (ignoring leading/trailing space),
// and allow multiple languages.
bool getResp(const string &prompt, bool &val) {
string str;
if (! getResp(prompt, str, true)) return false;
const set<string> yes = {"yes", "y", "1", "si"};
const set<string> no = {"no", "n", "0", "nyet"};
if (yes.find(str) != yes.end()) {
val = true;
return true;
}
if (no.find(str) != no.end()) {
val = false;
return true;
}
return false;
}
// Test driver for your situation.
int main() {
unsigned long birthYear;
std::string dogBreed;
bool isVaccinated;
if (! getResp("What year was the dog born? ", birthYear)) {
std::cout << "** ERROR, invalid value\n";
return 1;
}
if (! getResp("What is the breed of the dog? ", dogBreed, true)) {
std::cout << "** ERROR, invalid value\n";
return 1;
}
if (! getResp("Has the dog been vaccinated? ", isVaccinated)) {
std::cout << "** ERROR, invalid value\n";
return 1;
}
std::cout
<< birthYear
<< " '" << dogBreed << "' "
<< (isVaccinated ? "yes" : "no") << '\n';
}