• 使用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()使得检查错误更简单些。可以用其中的任何一个来读取一行输入;只是应该知道,它们的行为稍有不同。

  • 相关阅读:
    Apache Phoenix系列 | 从入门到精通(转载)
    Phoenix 简单介绍
    ES 调优查询亿级数据毫秒级返回!怎么做到的?--文件系统缓存
    Linux 文件系统缓存 -针对不同数据库有不同作用
    Hive 调优
    clickhouse 中文论坛
    从0到N建立高性价比的大数据平台(转载)
    ClickHouse 分布式高可用集群搭建(转载)
    Hive 模式设计
    Oracle 分区表中本地索引和全局索引的适用场景
  • 原文地址:https://www.cnblogs.com/MuSmile/p/5452891.html
Copyright © 2020-2023  润新知