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