C++心得笔记
C++ Primer Plus 第四章 复合类型
4.2.3 字符串输入
下面程序揭示了程序strings.cpp的一个缺陷,这种能通过精心选择输入被掩盖掉。
#include<iostream>
int main()
{
using namespace std;
const int ArSize = 20;
char name[ArSize];
char dessert[ArSize];
cout << "Enter your name:\n";
cin >> name;
cout << "Enter your favorite dessert:\n";
cin >> dessert;
cout << "I have some delicious " << dessert;
cout << " for you, " << name << ".\n";
return 0;
}
该程序需要用户连续输入名字和点心两个字符串,当输入的名字是两个单词加一个空格分离时,程序会发生一下情况:
我们还没有看到提示我们输入点心,程序就把它显示了出来,然后立即显示最后一行。
这是为什么呢?
cin使用空白(空格、制表符和换行符)来确定字符串的结束位置 ,这意味着cin在获取字符串数组输入时只读取一个单词。读取该单词后,cin将该字符串放到数组中,并自动在结尾添加空字符。
4.2.4 每次读取一行字符串输入
为解决上面那种情况,可以采用另一种字符串读取方法。具体地说采用面向行而不是面向单词的方法。iosream中的类(如cin)提供了一些面向行的类成员函数:getline()和get()。这两个函数都读取一行输入,直到到达换行符。
getline()和get()函数的区别在于,getline()在读取完一行后,将丢弃换行符,而get()将换行符保留在输入序列中。
1、面向行的输入:getline()
getline()函数读取整行,它使用通过回车键输入的换行符来确定输入结尾。要调用这种方法,可以使用cin.getline()。该函数有两个参数。第一个参数是用来存储输入行的数组的名称,第二个参数是要读取的字符数。如果这个参数为20,则函数最多读取19个字符,余下的空间用于存储自动在结尾处添加的空字符。getline()成员函数在读取指定数目的字符或遇到换行符时停止读取。
例如:
cin.getline(name,20);
下面程序用cin.getline()修改了之前的程序:
#include<iostream>
int main()
{
using namespace std;
const int ArSize = 20;
char name[ArSize];
char dessert[ArSize];
cout << "Enter your name:\n";
cin.getline(name, ArSize);
cout << "Enter your favorite dessert:\n";
cin.getline(dessert, ArSize);
cout << "I have some delicious " << dessert;
cout << " for you, " << name << ".\n";
return 0;
}
程序输出:
2、面向行的输入:get()
iostream类有另一个名为get()的成员函数,该函数有几种变体。其中一种变体的工作方式与getline()类似,它们接受的参数相同,解释参数的方式也相同,并且都读取到行尾。但get()并不再读取并丢弃换行符,而是将其留在输入队伍中。假设我们连续两次调用get():
cin.get(name, ArSize);
cin.get(dessert, ArSize);
由于第一次调用后,换行符将留在输入队列中,因此第二次调用时看到的第一个字符便是换行符。因此get()认为已到达行尾,而没有发现任何可读取的内容。如果不借助于帮助,get()将不能跨过该换行符。
幸运的是,get()有另一种变体。使用不带任何参数的cin.get()调用可读取下一个字符(即使是换行符),因此可以用它来处理换行符,为读取下一行输入做好准备。也就是说,可以采用下面的调用序列:
cin.get(name, ArSize);
cin.get();
cin.get(dessert, ArSize);
另一种使用get()的方法是将两个类成员函数拼接起来(合并),如下图所示:
cin.get(name, ArSize).get(); // concatenate member functions
#include<iostream>
int main()
{
using namespace std;
const int ArSize = 20;
char name[ArSize];
char dessert[ArSize];
cout << "Enter your name:\n";
cin.get(name, ArSize).get();
cout << "Enter your favorite dessert:\n";
cin.get(dessert, ArSize).get();
cout << "I have some delicious " << dessert;
cout << " for you, " << name << ".\n";
return 0;
}
为什么使用ger(),而不是getline()呢?首先,老式实现没有getline()。其次,get()使输入更仔细。
4.2.5 混合输入字符串和数字
混合输入数字和面向行的字符串会导致问题。例如下面程序:
#include<iostream>
int main()
{
using namespace std;
cout << "what year was your house built>\n";
int year;
cin >> year;
cout << "what is its street adress?\n";
char address[80];
cin.getline(address, 80);
cout << "year bult: " << year << endl;
cout << "address: " << address << endl;
cout << "Done!\n";
return 0;
}
程序输出:
用户根本没有输入地址的机会。问题在于,输入数字时的换行符留在输入队列中,后面的cin.getline()看到换行符后,认为是一个空行,并将一个空字符赋给address数组。解决方式是,在读取地址之前先读取并丢弃换行符。例如:
cin >> year;
cin.get();
或者利用表达式cin>>year返回cin对象,将调用拼接起来:
(cin >> year).get(); // or(cin >> year).get(ch)