使用getline和get方法读取字符串

使用cin对象对C风格字符串执行输入操作时存在一个缺陷,如下:

#include <iostream>
using namespace std;

int main()
{
    const int stringSize = 64;

    char string1[stringSize];
    char string2[stringSize];
    
// enter 1st string cout
<< "Enter first string: "; cin >> string1;
// enter 2nd string cout
<< "Enter second string: "; cin >> string2;
  // show the result of input cout
<< "The first string is " << "\"" << string1 << "\"" << ", the second string is " << "\"" << string2 << "\"" << endl;
  // pause
  system("pause");
return 0; }

一般情况下该程序应该可以正常完成工作:接收用户输入的两段字符串(有长度限制),并一起进行输出。

然而当用户的输入中包含空格等空白元素时,则会出现下述意料之外的状况

Enter first string: test string1
Enter second string: The first string is "test", the second string is "string1"

 

解释上面的运行情况之前,不妨先考虑一个问题,cin对象是如何确定已完成字符串输入?由于C风格字符串使用空字符‘0’作为结尾,而这种空字符是无法通过键盘之间输入的,因此cin需要借助别的方法来确定字符串的结尾位置。cin使用空白(空格,制表符,换行符)来确定字符串的结尾位置,这意味着cin在获取字符数组输入时只读取一个单词(或者说不内含空白的一段字符串)。读取该单词后,cin将该字符串放到数组中,并自动在结尾添加空字符。这样一来也就能很好的解释上面的情况了。


面向行的输入

  每次读取一个单词通常不是最好的选择,要将整条短语而不是一个单词作为字符串输入,需要采用另一种字符串读取方法。具体的说,需要采用面向行而不是面向单词的方法。幸运的是,istream中的类提供了一些面向行的类成员函数:getline()和get()。这两个函数都读取一行输入,直到到达换行符。然而,随后getline()将丢弃换行符,而get()将换行符保留在输入序列中,下面详细介绍它们,首先介绍getline()。

  • cin.getline()

  getline()函数读取整行,它使用通过回车键输入的换行符来确定输入结尾。要调用这种方法,可以使用cin.getline()。该函数有两个参数。第一个参数是用来存储输入行的数组的名称,第二个参数是要读取的字符数。如果这个参数为20,则函数最多读取19个字符,余下的空间用于存储自动在结尾处添加的空字符。getline()成员函数在读取指定数目的字符或遇到换行符时停止读取。

#include <iostream>
using namespace std;

int main()
{
    const int stringSize = 64;

    char string1[stringSize];
    char string2[stringSize];
    
    // enter 1st string
    cout << "Enter first string: ";
    //cin >> string1;
    cin.getline(string1, stringSize);

    // enter 2nd string
    cout << "Enter second string: ";
    //cin >> string2;
    cin.getline(string2, stringSize);

    // show the result of input
    cout << "The first string is " << "\"" << string1 << "\""
        << ", the second string is " << "\"" << string2 << "\"" << endl;

    // pause
    system("pause");
    return 0;
}

  运行结果:

  Enter first string: test string1
  Enter second string: test string2
  The first string is "test string1", the second string is "test string2"

  • cin.get()

  我们来试试另一种方法。istream类有另一个名为get()的成员函数,该函数有几种变体。其中一种变体的工作方法与getline()类似,它们接受的参数相同,解释参数的方式也相同,并且都读取到行尾,但get()并不再读取并丢弃换行符,而是将其留在输入队列中。假设我们连续两次调用get():

  cin.get(string1, stringSize);

  cin.get(string2, stringSize);  // 将会出现问题

  由于第一次调用后,换行符将留在输入队列中,因此第二次调用时看到的第一个字符便是换行符,因此get()认为已经到达行尾,而没有发现任何可读取的内容。如果不借助帮助,get()将不能跨过该换行符。

  幸运的是,get()有另一种变体。使用不带任何参数的cin.get()调用可读取下一个字符(即使是换行符),因此可以用它来处理换行符,为读取下一行输入做好准备。也就是说,可以采用下面的调用序列:

  cin.get(string1, stringSize);  // 读取第一行

  cin.get();               // 读取换行符

  cin.get(string2, stringSize);  // 读取第二行

  另一种使用get()的方式是将两个类成员函数拼接起来(合并),如下所示:

  cin.get(string1, stringSize).get();

  之所以可以这样做,是由于cin.get(string1, stringSize)返回一个cin对象,该对象随后将被用来调用cin.get()函数。同样,下面的语句将把输入中连续的两行分别读入到数组string1和string2中,其效果与两次调用cin.getline()相同:

  cin.getline(string1, stringSize).getline(string2, stringSize);

#include <iostream>
using namespace std;

int main()
{
    const int stringSize = 64;

    char string1[stringSize];
    char string2[stringSize];
    
    // enter 1st string
    cout << "Enter first string: ";
    //cin >> string1;
    cin.get(string1, stringSize).get();

    // enter 2nd string
    cout << "Enter second string: ";
    //cin >> string2;
    cin.get(string2, stringSize).get();

    // show the result of input
    cout << "The first string is " << "\"" << string1 << "\""
        << ", the second string is " << "\"" << string2 << "\"" << endl;

    // pause
    system("pause");
    return 0;
}

  运行结果:

  Enter first string: test string1
  Enter second string: test string2
  The first string is "test string1", the second string is "test string2"

 

  为什么要使用get(),而不是getline()呢?首先,老式实现中并没有getline()。其次,get()使输入更仔细。例如,假设用get()将一行读入数组中。如何知道停止读取的原因是由于已经读取了整行,而不是由于数组已经被填满?查看下一个输入字符,如果是换行符,说明已读取了整行;否则,说明该行中还有其他输入。总之,getline()使用起来简单一些,但get()使得检查错误更简单些。可以用其中的任何一个来读取一行输入;只是应该知道,它们的行为稍有不同。

 

posted @ 2016-05-05 14:36  慕Smile  阅读(2616)  评论(0编辑  收藏  举报