文章目录
C++ | string类详解
1、标准库中的string类
1.1string类介绍
略过,可以参考String类
1.2auto关键字和范围for读写string
例:
#include<iostream>
using namespace std;
int main()
{
string str = "abcdefgh";
for (auto ch : str) {//因为知道类型是char,所以也可以将auto改成char
cout << ch << ' ';
}//运行结果:`a b c d e f g h`
cout << endl;
for (auto ch : str) {
cout << ++ch << ' ';
}//运行结果:`b c d e f g h i`
return 0;
}
1.2.1 auto关键字
这里补充2个C++11的小语法
- 在早期C/C++中auto的含义是:使用**auto修饰的变量,是具有自动存储器的局部变量,后来这个不重要了。**C++11中,标准委员会变废为宝赋予了auto全新的含义即:auto不再是一个存储类型指示符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。
- 用auto声明指针类型时,用auto和auto*没有任何区别,但用auto声明引用类型时则必须加&
- auto不能作为函数的参数,可以做返回值,但是建议谨慎使用
- auto不能直接用来声明数组
auto i = 5; //推断为int类型 auto str = "hello world"; //推断为const char*类型
auto mysum(int a1, int a2)->int { return a1 + a2; }//返回类型后置,auto为返回值占位符,真正的返回类型是int
- 当在同一行声明多个变量时,这些变量必须是相同的类型,否则编译器将会报错,因为编译器实际只对第一个类型进行推导,然后用推导出来的类型定义其他变量,简单来说就是从左往右的推导
auto x= 1, y = 4.2;
报错,因为先推导x的类型为int,再推导y的类型为double,但是x和y的类型不一样,所以会报错
- auto会使用表达能力更强的类型、
auto x = true ? 1 : 4.2;
在标准中,auto会采用表达能力更强的类型,所以x的类型会推导为double类型
1.2.2范围for
组成内容:
for循环后的括号由冒号
:
分为两个部分:
范围声明式:范围内用于迭代的变量,其类型式范围表达式中元素的类型或元素的类型的引用
范围表达式:表示被迭代的范围,可以是数组或者式对象,对于对象必须满足以下之一的条件:
- 对象类型定义了begin和end成员函数
- 定义了以对象类型为参数的begin和end普通函数。
特点:
- 自动迭代,自动取数据,自动判断结束。
- 范围for可以作用到数组和容器的对象上进行遍历
- 范围for的底层很简单,容器的遍历实际上就是替换为迭代器,这个从汇编层也可以看到
举例:
#include<iostream>
#include <string>
#include <map>
using namespace std;
int main()
{
int array[] = { 1, 2, 3, 4, 5 };
// C++98的遍历
for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
{
array[i] *= 2;
}
for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
{
cout << array[i] << endl;
}
// C++11的遍历
for (auto& e : array)
e *= 2;
for (auto e : array)
cout << e << " " << endl;
string str("hello world");
for (auto ch : str)
{
cout << ch << " ";
}
cout << endl;
return 0;
}
1.3string类的常用接口说明
1.3.1 常见构造方式
函数名称 | 功能说明 |
---|---|
string() | 构造空的string类对象,即空字符串 |
string(const char*s) | 用string来构造string类对象 |
string(size_t,char c) | 包含n个字符c |
string(const string&s) | 拷贝构造函数 |
举例:
int main()
{
string s1; // 构造空的string类对象s1
string s2("hello world"); // s2:helloc world 用C格式字符串构造string类对象s2
string s3(5,c); // s3:ccccc 用五个c来初始化s3
string s4(s2); // s4:hello world 拷贝构造s3
return 0;
}
1.3.2 常见容量相关操作
函数名称 | 功能 |
---|---|
size | 返回字符串有效字符长度 |
length | 返回字符串有效字符长度 |
capacity | 返回空间总大小 |
empty | 检测字符串释放为空串,是然会true,否则返回false |
clear | 清空有效字符 |
reserve | 为字符串预留空间 |
resize | 将有效字符的个数改成n个,多出的空间用字符c来填充 |
举例:
int main()
{
string s1 = "hello world";
string s2("hello string");
cout << s1.size();
cout << s1.length();
cout << s1.capacity();
cout <<
return 0;
}
注意:
size()
与length()
方法底层实现原理完全相同,引入size()
的原因是为了与其他容器的接口保持一致,一般情况下基本都是用size()
8880clear()
只是将string中的有效字符清空,不改变底层空间大小。resize(size_t n)
与resize(size_t n, char c)
都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(n)
用0来填充多出的元素空间,resize(size_t n, char c)
用字符c来填充多出来的元素空间。注意:resize
在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。reserve(size_t res_arg = 0)
:为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserver不会改变容器的大小。
1.3.3 string类对象的访问及遍历操作
函数名称 | 功能说明 |
---|---|
operator[] | 返回pos位置的字符,const string 类对象调用 |
begin + end | begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器 |
rbegin + rend | begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器 |
范围for | C++11支持更简洁的范围for的新遍历方式 |
1.3.4 string类对象的修改操作
函数名称 | 功能 |
---|---|
push_back | 在字符串后面尾插字符 |
append | 在字符串后追加一个字符串 |
operator+= | 在字符串后追加字符串 |
c_str | 返回c格式字符串 |
find + npos | 从字符串指定位置开始往后找字符,返回该字符在字符串中的位置 |
rfind | 从字符串指定位置开始往前找字符,返回该字符在字符串中的位置 |
substr | 在str中从指定位置开始,截取n个字符,然后将其返回 |
注意:
- 在string尾部追加字符时,
s.push_back(c) ; s.append(1,c) ; s += 'c'
三种的实现方式差不多,一般情况下string类的+=操作用的比较多,常常用于连接字符或者字符串- 对string类对象操作时,如果能够大概预估到放多少字符,可以先通过
reserve
把空间预留好。
1.3.5 string类非成员函数
函数 | 功能说明 |
---|---|
operator+ | 由于传值返回,导致深拷贝效率低 |
operator>> | 输入运算符重载 |
operator<< | 输出运算符重载 |
getline | 获取一行字符串 |
relational operators | 大小比较 |
输入运算符重载 |
| operator<<
| 输出运算符重载 |
| getline
| 获取一行字符串 |
| relational operators
| 大小比较 |