17

基本上我首先将一个整数作为输入,然后是测试用例。我的每个测试用例都是一个字符串。如果字符串的起始模式与“HI A”匹配并且不区分大小写,我想将字符串打印回来。我写了下面的代码来完成这个。我的问题是,当我在每次输入后按 enter 时,getline 将换行符作为新输入。我试图通过在每次输入后使用额外的 getline 来解决这个问题,但问题仍然存在。即使我设置了中断条件,程序也会卡在循环中。我究竟做错了什么?

#include <iostream>
#include <string>
using namespace std;
int main(){
    int N;
    cin >>N;
    string nl;
    getline(cin,nl);
    for (int i=0;i<N;i++){
        string s;
        getline(cin,s);
        //cout <<"string"<<s<<endl;
        int flag=0;
        if ((s.at(0)=='h'||s.at(0)=='H')&&(s.at(1)=='i'||s.at(1)=='I')&&(s.at(2)==' ')&&(s.at(3)=='a'||s.at(3)=='A')) flag=1;

        if (flag==1) cout << s;
        //cout << "not " <<s;
        string ne;
        cout << "i="<< i<<endl;
        if (i==N-1) {break;}
        getline(cin,ne);

    }
}

这是示例输入:

5
Hi Alex how are you doing
hI dave how are you doing
Good by Alex
hidden agenda
Alex greeted Martha by saying Hi Martha

输出应该是:

Hi Alex how are you doing
4

7 回答 7

14

ignore() 函数可以解决问题。默认情况下,它会丢弃所有输入序列,直到换行符为止。

也可以指定其他分隔符和字符限制。

http://www.cplusplus.com/reference/istream/istream/ignore/

在你的情况下,它是这样的。

    cin >> N;
    cin.ignore();
于 2016-08-16T04:22:28.953 回答
9

cin >>N停在第一个非数字字符,即换行符处。这个你必须getline读过去,这很好。

之后的每个附加getline内容都会读取整行,包括末尾的换行符。通过投入一秒钟getline,您将跳过一半的输入。

于 2013-09-10T17:50:27.517 回答
2

所以,你真正的问题不是getline吃换行符,而是你的第二个问题是吃换行符getline(cin, ne)......

那是因为你错误地认为你需要两个getline操作来读取一行 - 或者类似的东西。混合“linebased”和“itembased”输入确实有令人困惑的方式来处理换行符,所以你确实需要一些东西来“跳过” frin 后面留下的换行符cin >> N;,但是一旦你摆脱了它,你只需要 ONEgetline来阅读和包括行尾的换行符。

于 2013-09-10T17:45:47.950 回答
1

你只需要接受 getline 最后会给你 '\n' 的事实。一种解决方案是在获取后删除 '\n'。另一种解决方案是不要写额外的'endl'。例如,对于您的问题,您可以使用此代码

int N;
cin >> N;
string line;
getline(cin, line); // skip the first new line after N.
for (int i = 0; i < N; i++) {
  string line;
  getline(cin, line);
  string first4 = line.substr(0, 4);
  // convert to upper case.
  std::transform(first4.begin(), first4.end(), first4.begin(), std::ptr_fun<int, int>(std::toupper)); // see http://en.cppreference.com/w/cpp/algorithm/transform
  if (first4 == "HI A") {
    cout << line;  // do not include "<< endl"
  }
}
于 2013-09-10T17:52:40.993 回答
1

我正在写这个答案,希望它可以帮助那些想要一个非常简单的解决这个问题的人。

就我而言,问题是由于某些文件具有不同的行尾,例如“\r”与“\n”。在 Windows 中一切正常,但在 Linux 中却失败了。

答案其实很简单。在读入每一行后,我创建了一个函数 removeNewLineChar。这样就删除了字符。removeNewLineChar 接收读入的行并将其逐个字符地复制到新字符串中,但它避免复制任何一个换行符。

这是对其工作原理的完整说明。

C++ getline 读取换行符以及如何避免它

于 2019-02-14T03:20:37.450 回答
0
std::string line;
std::cin>>std::ws; // discard new line not processed by cin
std::getline(std::cin,line);

来自注释部分https://en.cppreference.com/w/cpp/string/basic_string/getline

When consuming whitespace-delimited input (e.g. int n; std::cin >> n;) any whitespace that follows, including a newline character, will be left on the input stream. Then when switching to line-oriented input, the first line retrieved with getline will be just that whitespace. In the likely case that this is unwanted behaviour, possible solutions include:

An explicit extraneous initial call to getline
Removing consecutive whitespace with std::cin >> std::ws
Ignoring all leftover characters on the line of input with cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n'); 
于 2021-09-22T16:33:21.753 回答
0

cin.ignore()为我工作。

void House::provideRoomName()
{
    int noOfRooms;

    cout<<"Enter the number of Rooms::";
    cin>>noOfRooms;
    cout<<endl;

    cout<<"Enter name of the Rooms::"<<endl;
    cin.ignore();
    for(int i=1; i<=noOfRooms; i++)
    {
        std::string l_roomName;
        cout<<"Room"<<"["<<i<<"] Name::";
        std::getline(std::cin, l_roomName);
    }
}
于 2018-11-03T15:58:27.160 回答