C++基础之字符串string
标准库类型string
表示可变长的字符序列,使用string
类型必须首先包含string
头文件。作为标准裤的一部分,string
定义在命名空间std
中。
定义和初始化string对象
初始化string对象
的方法有很多种,具体可以参见如下的表格:
方法 | 含义 |
---|---|
string str | 默认初始化,str是一个空串 |
string str2(str1) | str2是str1的副本 |
string str2 = str1 | 等价于str2(str1),str2时str1的副本 |
string str3("value") | str3是字面值"value"的副本,除了字面值字符串最后的结束字符 |
string str3 = "value" | 同str3("value"),str3是"value"的副本 |
string str4(n, 'c') | 把str4初始化为由n个字符c组成的串 |
string对象上的操作
如下列出的是string对象上的大部分操作:
方法 | 含义 |
---|---|
str.empty() | 判断是否为空 |
str.size() | 返回str中字符的个数 |
str1 + str2 | 返回str1和str2连接的结果 |
str1 = str2 | 用str2的副本代替str1中原来的字符 |
==, !=, <, <=, >, >= | string对象的相等性判断,对大小写敏感 |
读写string对象
我们知道可以使用标准库中的iostream
来读写int
, double
等内置类型的值。同样的,我们也可以使用IO操作符读写string对象。
#include<iostream>
#include<string>
using namespace std;
int main() {
string s; // 空字符串
cin >> s; // 将string对象读入s,遇到空白停止
cout << s << endl; // 输出s
return 0;
}
在以上的代码段中,is>>s
读入操作符在遇到空格的时候会自动停止,而且会忽略开头的空白从第一个真正的字符开始读起。如果要读入Hello world!
这样的内容,则可以这样做:
string s1, s2;
cin >> s1 >> s2; // 读入第一个到s1中再读入第二个到s2中
cout << s1 << s2 << endl;
读取数量未知的string对象
通过对输入流的判断得知当前的读取是否有效
int main() {
string word;
while (cin >> word)
cout << word << endl;
return 0;
}
使用getline读取一整行
有时候我们想要能够连续的读入多个单词到一个string对象中,即使遇到了空格也能够继续的读下去,这个时候就应该是getline
函数代替原来的is>>s
操作符。
getline
函数的参数是一个输入流和一个string对象,函数从输入流中读入内容,直到换行符为止(注意换行符也被读了进来),然后把所读的内容存入到string对象中(注意不存换行符)。如果一开始输入的便是换行符,那么所得的结果就是空的string。
int main() {
string line;
while (getline(cin, line)) {
cout << line << endl;
}
return 0;
}
- 因为line中不包含换行符,所以需要手动的加上换行符。
- 触发getline函数返回的换行符实际被丢失了,实际得到的string对象不包含换行符。
string的size操作
size
函数返回string对象的长度,即string对象中字符的个数,其类型为string::size_type
。
string类及其他大多数标准库类型都定义了几种配套的类型,这些配套的类型体现了标准库类型与机器无关的特性。
string::size_type
是一个无符号类型的值而且能够存放下任何string对象的大小。因为是无符号数所以不能和有符号数混用。例如,假设n是一个具有负值的int,则表达式s.size() < n
的判断结果几乎肯定是true
,这是因为负数会自动的转换成一个比较大的无符号数。
如果一条表达式中已经有了
size()
函数就不要再使用int了,这样就可以避免上述的意外发生。
我们可以使用auto类型
或decltype
来使用string::size_type
类型的值。
auto len = line.size();
字面值和string对象相加
我们知道两个string对象通过相加会得到一个新的string对象,其内容就是把后一个对象与前一个对象拼接而成。不仅如此,标准库还允许把字符字面值和字符串字面值转换成string对象与string对象进行相加,所以我们能够写出这样的代码:
string s1 = "hello", s2 = "world";
string s3 = s1 + ", " + s2 + '
'; // string对象和字符串字面值和自负字面值相加
但是要注意的是必须确保每个加法运算符的两侧运算对象至少有一个string对象。
string s4 = s1 + ", "; // 正确,一个string对象和一个字面值相加
string s5 = "hello" + ", "; // 错误,两个运算对象都不是string
string s6 = s1 + ", " + "world"; // 正确,每个运算符两侧都是string
string s7 = "hello" + ", " + s2; // 错误,第一个运算符两侧都是字面值
处理string对象中的字符
在这里先提个题外话,在C++的cctype头文件中定义了一组标准库函数来处理某个字符的特性,具体见表:
操作 | 含义 |
---|---|
isalnum(c) | 当c是字母或是数字时为真 |
isalpha(c) | 当c是字母时为真 |
isdigit(c) | 当c是数字时为真 |
islower(c) | 当c是小写字母时为真 |
isupper(c) | 当c是大写字母时为真 |
tolower(c) | 如果c是大写字母,则输出对应的小写字母;否则原样输出c |
toupper(c) | 如果c是小写字母,则输出对应的大写字母;否则原样输出c |
C++标准库中除了定义C++语言特有的功能外,也兼容C语言的标准库。C语言的头文件形如name.h,C++则将这些文件命名为cname。去掉了
.h
后缀,添加了前缀c
,表示这是一个属于C语言标准库的头文件。一般来说,C++程序应该使用名为cname的头文件而不是用name.h的形式,标准库中的名字总能在命名空间
std
中找到。
使用范围for语句处理每个字符
通过使用范围for
语句,我们可以很方便的遍历整个字符串。其语法形式为:
for (declaration : expression)
statement
其中, expression部分是一个对象,用于表示一个序列。declaration部分负责定义一个变量,该变量将被用于访问序列中的基础元素。每次迭代,declaration部分的变量会被初始化为expression部分的下一个元素值。
通过使用范围for
语句可以遍历string对象中的每个字符。
string str("some string");
for (auto c : str) {
cout << c << endl;
}
还可以通过引用改变字符串中的字符:
string str("some string");
for (auto &c : str) {
c = toupper(c);
}
cout << str << endl;
除了使用引用的方法外还可以通过下标来处理字符:
for (decltype(str.size()) index = 0; index != str.size(); ++index) {
str[index] = toupper(str[index]); // 将当前字符改写为大写形式
}
可以看到c++中的字符串对象要比c中的字符串数组好用的多。