C ++和cin.getline问题

时间:2012-12-24 08:44:09

标签: c++ struct getline

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

如果我在struct中使用char,它只会复制我的第一个单词,如果我试图使用字符串,它会跳过第一个问题(忽略它)。

你能指导我吗?

稍后编辑: 我使用Microsoft Visual C ++ 2010快速版,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 ve-ti 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();
}

1 个答案:

答案 0 :(得分:2)

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

  • void main()是非标准的。 main()始终返回intmain()的唯一可移植声明为int main()int main(int argc, char* argv[])(或int main(int argc, char** argv),与前一个相同;当然,这两个论点的名称可以自由选择。)
  • 您使用的get()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);

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

(std::cin >> std::ws).get(pointer[i].artist, 50);

操纵器 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);

那就是说,我猜你真的不想使用get(),因为这个函数在字符串中包含了分隔字符(默认为'\n')。您可能希望使用getline()代替。此外,更容易处理std::string而不是字符数组,即我会对struct使用不同的定义:

struct melodi
{
    std::string artist;
    std::string titlu;
    int an;
    int lungime;
};

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

std::getline(std::cin >> skipline, artist);