我整个星期都忙于这项作业。当我让程序最终运行时,我意识到仅使用String
,如果我的输入有空格,就会破坏代码(因为我的程序需要收集3个单独的变量,首先是一个int,然后是一个字符串,最后一个布尔)。因为这是第二个变量,所以使用带有白色字符的短语会使我的代码混乱。当我尝试将其更改为cin >> breed
或cin.get
时,这是我收到的错误消息:
c2664错误“无法将参数1从std :: string转换为_Elem *”
下面是有问题的代码(中间一行显示错误)。任何帮助将不胜感激!
cin.getline
答案 0 :(得分:1)
首先,您需要了解C ++中有两 getline
件事,一个在I / O区域,一个在顶级标准名称空间。 >
cin.getline(breed, 100)
是I / O区域中的那个(特别是istream::getline()
,它对字符串一无所知,而是更喜欢使用字符数组。您应该避免这种情况一个。
了解的字符串是std::getline()
,如果您不想回到C遗留的“字符串”的糟糕年代,那通常是首选
此外,在混合使用特定于类型的输入(如<<
)和特定于行的输入(如getline
)时,在C ++中需要小心。了解每个操作前后文件指针的位置很重要。
例如,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';
}
您还可以选择确保 all 输入是基于行的(输入后将这些行转换为正确的类型),因为这可能会简化确保指针位于正确的位置,并且输入中的错误(例如输入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';
}