C++初学(7)

7.1、字符串

字符串是存储在内存的连续字节中的一系列字符。C++处理字符串的方式有两种,第一种是来自C语言,常被称为C风格字符串,另一种则是基于string类库的方法。

存储在连续字节中的一系列字符意味着可以将字符存储在char数组中,其中每个字符都位于自己的数组元素只能中。C风格字符串以空字符结尾,空字符被写作\0,其ASCLL码为0,用来标记字符串的结尾。

char dog[8]={'b','e','a','u','x',' ',''I','I'};
char cat[8]={'f','a','t','e','s','s','a','\0'};

如上面两行代码,只有第二个是字符串。cout等函数都逐个处理字符串中的字符,直到空字符为止。如果用cout显示cat数组时,则将显示前7个字符;如果用cout显示dog数组时,则将显示数组内的8个字母,并接着将内存中随后的各个字符解释为要打印的字符,直到遇到空字符位置。

将数组初始化字符串不必像上面那么复杂,可以只需使用一个用引号括起来的字符串即可,这种字符串被称为字符串常量或字符串字面量。如下所示:

char bird[11]="Mr.Chess";
char chicken[]="CXKuns";

用引号括起来的字符串包括结尾的空字符,因此不用写。如果通过键盘进行输入字符串时,将会自动加上结尾的空字符,因此要确保数组足够大能够存放空字符。

注:字符串常量(使用双引号)不能与字符常量(使用单引号)互换。如'S'只是83的另一种写法,而"S"表示的是两个字符(字符S和\0)。

7.1.1、拼接字符串常量

有时候字符串很长不能放到一行,C++允许进行拼接。如下面语句都是等效的:

cout<<"I'd give my right arm to be " "a great violinist.\n";
cout<<"I'd give my right arm to br a great violinist.\n";
cout<<"I'd give my right ar"
"m to be a great violinist.\n";

7.1.2、在数组中使用字符串

下面给一个程序,将一个数组初始化为用引号括起的字符串,并使用cin将一个输入字符串放到另一个数组中。还使用了C语言库函数strlen()来确定字符串的长度。头文件是cstring(老式为string.h)。

#include <iostream>
#include <cstring>
int main()
{
	using namespace std;
	const int Size = 15;
	char name1[Size];
	char name2[Size] = "C++程序员";

	cout << "Hello!I'm " << name2;
	cout << "!What's your name?" << endl;
	cin >> name1;
	cout << "Well, " << name1 << ",your name has ";
	cout << strlen(name1) << " letters and is stored" << endl;
	cout << "in an array of " << sizeof(name1) << " bytes." << endl;
	cout << "Your initial is " << name1[0] << ".\n";
	name2[3] = '\0';
	cout << "Here are first 3 characters of my name: ";
	cout << name2 << endl;
	return 0;
}

程序说明:

(1)sizeof运算符指出整个数组的长度:15字节;

(2)strlen()函数返回的是存储在数组中的字符串的长度,而不是数组本身的长度,另外strlen()只计算可见的字符,不计算空字符,因此对于Furina返回的值是6。

(3)由于name1和name2是数组,所以可以用索引来访问数组中各个字符。另外程序将name2[3]设置为空字符,这使得字符串在第3个字符后即结束。

7.1.3、字符串输入

下面给一个程序

#include <iostream>
int main()
{
	using namespace std;
	const int ArSize = 20;
	char name[ArSize];
	char dessert[ArSize];

	cout << "Enter your name:" << endl;
	cin >> name;
	cout << "Enter your favorite dessert:" << endl;
	cin >> dessert;
	cout << "I have some delicious " << dessert;
	cout << " for you," << name << ".\n";
	return 0;
}

哎?这个程序是不是有问题,怎么还没输入dessert就进行了下去?

由于不能通过键盘输入空租房有,因此cin需要用别的方法来确定字符串的结尾位置。cin使用空白(空格、制表符和换行符)来确定字符串的结束位置,这意味着cin在获取字符数组输入时只读取一个单词。读取该单词后,cin将该字符串放到数组中,并自动在结尾添加空字符。

这个程序中,cin把Alistair作为第一个字符串,并将他放到name数组中。把Dreeb留在输入队列中,当cin在输入队列搜索时,将Dreeb放进dessert数组中。

7.1.4、每次读取一行字符串输入

每次读取一个单词有时候很烦,如果要输入城市时将很难搞,这时候需要采用另一种字符串读取方式,采用面向行读取而不是单词读取。istream中的类提供了一些面向行的类成员函数:getline()和get()。这两个函数都读取一行输入,直到到达换行符。getline()将丢弃换行符,而get()将换行符保留在输入序列中。

7.1.4.1、getline()

getline()函数读取整行,它使用通过回车键输入的换行符来确定输入结尾。要调用这种方法,可以使用cin.getline()。该函数有两个参数,第一个参数是用来存储输入行的数组的名称,第二个参数是要读取的字符数。

例如,假设要试用getline()将姓名读入到一个包含20个元素的name数组中。可以这么调用:

cin.getline(name.20);

注:如果参数为20,那么函数只能最多读取19个字符,剩余空间用于存储自动在结尾处添加的空字符。

如果将上面程序修改为使用cin.getline(),将可以正常输入。

7.1.4.2、get()

istream类有一个名为get()的成员函数,该函数有几种变体,其中一种变体的工作方式与getline()类似,它们接受的参数相同,解释参数的方式相同,并且都读取到行尾。但get并不是停止读取且丢弃换行符,而是将换行符继续留在队列中。假设我们调用两次get(),程序只会运行第一次get(),因为第一次get()留下了换行符,第二次get()一开始读取到换行符,也就停止读取。

当然面对上面的情况下,get()有一种变体。使用不带参数的cin.get()调用可读取下一个字符(是的,换行符也可以),只要在调用两个get()之间加上cin.get()就行了。或者合并起来,如:

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

下面进行改进:

#include <iostream>
int main()
{
	using namespace std;
	const int ArSize = 20;
	char name[ArSize];
	char dessert[ArSize];

	cout << "Enter your name:" << endl;
	cin.get(name, ArSize).get();
	cout << "Enter your favorite dessert:" << endl;
	cin.get(dessert, ArSize).get();
	cout << "I have some delicious " << dessert;
	cout << " for you," << name << ".\n";
	return 0;
}

好了这样就能顺利实现了。

7.1.5、混合输入字符串和数字

混合输入数字和面向行的字符串会出现问题,看一下下面程序:

#include <iostream>
int main()
{
	using namespace std;
	cout << "What year your house bulit?" << endl;
	int year;
	cin >> year;
	cout << "What is its street address?" << endl;
	char address[80];
	cin.getline(address, 80);
	cout << "Year bulit: " << year << endl;
	cout << "Bulit: " << address << endl;
	cout << "Done!" << endl;
	return 0;
}

只输入了年份了就结束了运行,问题和上面一样,cin读取年份时,将回车键生成的换行符留在了输入队列里,因此将cin>>year改成(cin>>year).get()就行了。

#include <iostream>
int main()
{
	using namespace std;
	cout << "What year your house bulit?" << endl;
	int year;
	(cin >> year).get();
	cout << "What is its street address?" << endl;
	char address[80];
	cin.getline(address, 80);
	cout << "Year bulit: " << year << endl;
	cout << "Bulit: " << address << endl;
	cout << "Done!" << endl;
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值