0

我整个星期都被困在这个家庭作业上。就在我让程序最终运行时,我意识到通过使用 just cin >> breed,如果我的输入有空格,它会破坏代码(因为我的程序需要收集 3 个单独的变量,首先是一个 int,然后是一个字符串,最后是一个 bool) . 因为这是第二个变量,所以它使用带有白色字符的短语弄乱了我的代码。当我尝试将其更改为cin.getorcin.getline时,这是我收到的错误消息:

c2664 错误“无法将参数 1 从 std::string 转换为 _Elem *”

下面是有问题的代码(中间线给出了错误)。任何帮助将不胜感激!

#include <iostream>
#include <string>
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.getline(breed, 100);
    cin.ignore();
    cout << "Has the dog been vaccinated (1 = Yes/ 0 = No): ";
    cin >> vaccines;
}
4

1 回答 1

1

首先,您需要知道 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';
}
于 2019-06-13T03:08:03.763 回答