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;
}