“21天养成好习惯”第一期-21

12 篇文章 0 订阅

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)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值