C++ 和 cin.getline 的问题

Mir*_*hai 5 c++ struct getline

我正在学习structC++,但我不知道如何让它不跳过第一个问题。

如果我在其中使用 charstruct它只会复制我的第一个单词,如果我尝试使用 string 它会跳过第一个问题(忽略它)。

你能指导我吗?

后来编辑:我使用Microsoft Visual C ++ 2010 Express Edition的,并且是有之间的差异std::getlinecin.getline

#include <iostream>
#include <conio.h>
#include <string>


using namespace std;


struct melodie{

char artist[50];
char titlu[50];
int an;
int lungime;

};

void main(){
int repetam;
double rezultat;
cout<<"Cate melodii veti introduce: ";
cin>>repetam;
melodie *pointer = new melodie[repetam];
for(int i = 0; i<repetam; i++){
    cout<<endl;
    cout<<"Introduceti artistul melodiei: ";    
    cin.get(pointer[i].artist);
    cout<<"Introduceti titlul melodiei: ";
    cin.getline(pointer[i].titlu);
    cout<<"Introduceti anul melodiei: ";
    cin>>pointer[i].an;
    cout<<"Introuceti lungea melodiei (in secunde): ";
    cin>>pointer[i].lungime;


}
cout<<"Lista melodiilor introduse este:"<<endl;
for(int i =0; i<repetam; i++){
    cout<<"Artistul melodiei este: "<<pointer[i].artist<<endl
        <<"Titlul melodiei este: "<<pointer[i].titlu<<endl
        <<"Anul melodiei este: "<<pointer[i].an<<endl
        <<"Lungimea melodiei (in secunde) este: "<<pointer[i].lungime<<endl;
    cout<<endl;
}

cout<<"Melodiile care au aparut dupa anul 2000 si au lungimea mai mica de 180 de secunde sunt: "<<endl;
for(int i = 0;i<repetam; i++){
    if(pointer[i].an>2000 && pointer[i].lungime<180){
        cout<<"Artist: "<<pointer[i].artist
            <<endl
            <<"Titlul: "<<pointer[i].titlu<<endl<<endl;

        
    }
}



getch();
}
Run Code Online (Sandbox Code Playgroud)

Die*_*ühl 2

首先,你的could不能编译。有很多错误:

  • void main()是非标准的。main()总是返回 anint并且唯一的可移植声明是main()and int main()int main(int argc, char* argv[])或者int main(int argc, char** argv)与前一个相同;当然,两个参数的名称可以自由选择)。
  • get()您使用的and成员函数getline()至少带有一个附加参数:可用缓冲区的大小。

对于答案的其余部分,我假设这些错误已得到修复。

当然,输入并不是“跳过第一个问题”。相反,它读取已经给出的输入,但您可能没有这样考虑:格式化输入>>在其各自的格式完成(或失败)时立即停止读取。例如,当读取整数时,一旦遇到非数字,它就会停止。因此,当读取int输入repetam流的 时,当遇到用于完成此输入的换行符时,会在最后一个数字之后停止读取。调用get()读取一个字符串,直到(并包括)第一个换行符。输入时repetam输入换行符,因此,该字符串是get()停止的换行符。

更一般地说,当混合格式化输入(即使用>>)和未格式化输入(即使用命名函数get()getline()read()等)时,您通常需要删除不需要的字符。在您的示例中,您可能希望在调用之前删除第一个换行符get()(您需要包含<limits>才能访问std::numeric_limits):

std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
std::cin.get(pointer[i].artist, 50);
Run Code Online (Sandbox Code Playgroud)

如果您的artisti字符串不以任何空格字符开头,您可以使用

(std::cin >> std::ws).get(pointer[i].artist, 50);
Run Code Online (Sandbox Code Playgroud)

操纵 std::ws会跳过所有前导空格(它会跳过空格、制表符、换行符等),并且更容易键入。当然,如果您需要经常忽略换行符,您可以创建一个操纵器来忽略换行符之前的字符(包括换行符):

std::istream& skipline(std::istream& in)
{
    return in.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
}
// ...
(std::cin >> skipline).get(pointer[i].artist, 50);
Run Code Online (Sandbox Code Playgroud)

也就是说,我想您并不是真的想使用,因为该函数在字符串中get()包含分隔字符(默认情况下)。'\n'您可能想使用getline()它。另外,处理std::strings 而不是字符数组更容易,即我会使用不同的定义struct

struct melodi
{
    std::string artist;
    std::string titlu;
    int an;
    int lungime;
};
Run Code Online (Sandbox Code Playgroud)

这样,字符串可以是任意大小,并且不限于 49 个字符(您的版本中需要有空间用于终止空字符)。但是,要读取 astd::string您需要一个非成员函数):std::getline()

std::getline(std::cin >> skipline, artist);
Run Code Online (Sandbox Code Playgroud)