您的位置:首页 > 编程语言 > C语言/C++

关于c++中字符串的输入问题

2015-08-30 21:36 453 查看
转自:http://blog.sina.com.cn/s/blog_62f507e90100mm72.html

现看示例程序:

#include<iostream>

using namespace std;

int main()

{

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;

}

该程序的意图很简单:读取来自键盘的用户名和用户喜欢的甜点,然后显示这些信息。下面是程序的运行情况:

Enter your name:

Alistair Dreeb

Enter your favorite dessert:

I have some felicious Dreeb for you,Alistair.

我们甚至还没有对“输入甜点的提示”作出反应,程序便把它显示 出来了,然后立即显示最后一行。

cin是如何确定已完成字符串输入呢?由于不能通过键盘输入空值字符,因此cin需要用别的方法来确定字符串的结尾位置。cin使用空白--空格,制表符,和换行符--来定字符串的界。这意味着cin在获取字符数组输入时只读入一个单词。读取该单词后,cin将该字符串放到数组中,并自动在结尾添加空值字符。

这个例子的实际结果是,cin把Alistair作为第一个字符串,并把它放到name数组中。这把Dreeb留在输入队列中。当cin在输入队列中搜索用户喜欢的甜点时,它发现了Dreeb,因此cin读取Dreeb,并将它放到dessert数组中。

一 面向行的输入:getline() 和get()

为输入整条短语(而不是一个单词),并将它作为一个字符串,需要另一种字符串输入方法。具体地说,需要面向行的方法,而不是面向单词的方法。幸运的是,istream类(cin是其一个实例)有一些面向行的类成员函数。例如,getline()函数读取整行,它使用通过ENTER键输入的换行符来确定输入结尾。要调用这种方法,可以使用cin.getline()。该函数有两个参数。第一个参数是用来存储输入行的数组的名称,第二个参数是要读取的字符数。getline()成员函数在读取指定数目的字符或遇到换行符时停止读取。

修改上例程序如下:

#include<iostream>

using namespace std;

int main()

{

const int ArSize = 20;

char name[ArSize];

char dessert[ArSize];

cout<<"Enter your name:\n";

// cin>>name;

cin.getline(name,ArSize); //read through newline

cout<<"Enter your favorite dessert:\n";

// cin>>dessert;

cin.getline(dessert,ArSize);

cout<<"I have some delicious"<<dessert;

cout<<" for you ,"<<name<<".\n";

return 0;

}

下面是该程序的输出:

Enter your name:

Dirk Hammernose

Enter your favorite dessert:

Redish Torte

I have some deliciours Redish Torte for you ,Dirk Hammernose.

该程序现在就可以读取完整的姓名和用户喜欢的甜点了!getline()函数每次读取一行。它通过换行符来确定行尾,但并不保存换行符。相反,在存储字符串时,它用空值字符来替换换行符。

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

cin.get(name,ArSize);

cin.get(dessert,ArSize); // a problem

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

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

cin.get(name,ArSize);

cin.get();

cin.get(dessert,ArSize);

另一种方式是将两个类成员函数拼接起来,如下所示:

cin.get(name.ArSize).get();

之所以可以这么做,是由于cin.get(name,ArSize)返回一个cin对象,该对象随后被用来调用get()函数。同样,下面的语句:

cin.getline(name1.ArSize).getline(name2,ArSize);

将输入中连续的两行分别读入到数组name1,name2;其效果与两次调用cin.getline()相同。

下面是修改的程序:

#include<iostream>

using namespace std;

int main()

{

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;

}

二 空行和其他问题

当getline()或get()读取空行时,将发生什么情况呢?最初的做法是,下一条输入语句将在前一条getline()或get()结束读取的位置开始读取;但当前的做法是,当get()(不是getline())读取空行后将设置失效位。这意味着接下来的输入将会被阻断,但可以用下面的命令来恢复输入:

cin.clear()

另一个潜在的问题是,输入字符串可能比分配的空间长。如果输入行包含的字符串比指定的多,则getline()和get()将把剩下的字符留在输入队列中,而getline()还会设置失效位,并关闭后面的输入。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: