纵有疾风起,人生不言弃。本文篇幅较长,如有错误请不吝赐教,感谢支持。
💬文章目录
- 一.string容器基本概念
- 二.string容器常用操作
- ✅前言及函数参数的说明
- 一.构造和析构
- 二.string特性操作
- 三.字符操作
- 四.赋值操作
- 五.拼接操作
- 六.交换操作
- 七.string截取操作(子容器)
- 八.string的比较操作
- 九.string的查找替换
- 十.string的替换
- 十一.string插入操作(insert)
- 十二.string删除操作(erase)
一.string容器基本概念
C语言风格字符串(以空字符结尾的字符数组)太过复杂难于掌握,不适合大程序的开发,所以C++标准库定义了一种string类,定义在头文件。
string容器可以看做一片连续的储存空间,并用一个char*指向这片空间。string容器提供的迭代器是随机访问迭代器。
string和C风格字符串对比:
- ✅字符串是一个char * 类型指针,string是一个类string封装了char * ,用来管理这个字符串,是一个char*型的容器。
- ✅使用的时候,不必考虑内存分配和释放的问题: string管理char*所分配的内存。每一次string的复制,取值都由string类负责维护,不用担心复制越界和取值越界等。
- ✅动态管理内存(可扩展)
- ✅提供了大量操作容器的API(函数接口,查找find,拷贝copy,删除delete 替换replace,插入insert等)。缺点是效率略有降低,占用的资源也更多。
二.string容器常用操作
✅前言及函数参数的说明
本篇文章📃只对构造函数的使用进行了举例,其他函数的使用和构造函数的使用十分相似,万变不离其宗。
string容器可以看做一片连续的储存空间,有两个用途,一是用作字符串,二是用作存放数据的容器,string容器的成员函数有12个分类,前七个既可以用作字符串也可以用于存放数据的容器,而后五个(判断字符串是否相等、查找、替换、插入、删除)主要用于操作字符串,如果是其他格式的数据,这些函数也可以用,但意义不大。
关于函数参数的说明:
size_t:
可以理解为无符号整型(unsigned int)。
npos:
静态常量成员string::npos为字符数组的最大长度(通常为unsigned int的最大值)。
NBTS(null-terminated string):
C风格的字符串(以空字符0结束的字符串)。
一.构造和析构
💬string类有七个构造函数(C++11新增了两个)表格一览:
函数原型 | 解释 |
---|---|
string(); | 创建一个长度为0的string对象(默认构造函数) |
string(const char *s); | string对象初始化为s指向的NBTS(转换构造函数)。 |
string(const string &str); | 使用string对象str去初始化新创建的string对象(拷贝构造函数)。 |
string(const char *s,size_t n); | 将string对象初始化为s指向的地址后n字节的内容。 |
string(const string &str,size_t pos=0,size_t n=npos); | 将sring对象初始化为str从位置pos开始到结尾的字符(或从位置pos开始的n个字符) |
template string(T begin,T end); | 将string对象初始化为区间[begin,end]内的字符,其中begin和end是迭代器,其行为就像指针,用于指定位置,范围包括begin在内,但不包括end。 |
string(size_t n,char c); | 创建一个由n个字符c组成的string对象。析构函数~string()释放内存空间。 |
~string(); | 创建一个长度为0的string对象(默认构造函数)。 |
1)string();创建一个长度为0的string对象(默认构造函数)
#include <iostream>
#include<string>
using namespace std;
int main()
{
// 1)string():创建一个长度为0的string对象(默认构造函数)。
string s1; // 创建一个长度为0的string对象
cout << "s1=" << s1 << endl; // 将输出s1=,因为容器是空,所以输出是空。
cout << "s1.capacity()=" << s1.capacity() << endl; // 返回当前容量,可以存放字符的总数。
cout << "s1.size()=" << s1.size() << endl; // 返回容器中数据的大小。
}
string类的capacity方法,用于获取当前容器容量,意思是,如果不重新分配内存,所能存放字符的总数。
string类的size方法,用于获取容器中数据的大小,意思是里面已经存放了多少数据。
容器的当前容量是15,存放数据大小0,为什么当前容量不是0,而是15?
往string容器中存放数据时,如果数据多大就分配多少内存的空间,那么每次扩展容器内存时,都要重新分配和释放内存,效率很低,所以合理的做法就是预先分配比数据实际大小更多的空间。避免过于频繁的分配和释放内存。
还有,创建容器的时候,如果数据量小于15字节,就分配15字节,就算是空的,也分配15字节。
那我们再写几行代码,存放的数据,超过字节,这样的话,容器会做一次拓展。
int main()
{
// 1)string():创建一个长度为0的string对象(默认构造函数)。
string s1; // 创建一个长度为0的string对象
cout << "s1=" << s1 << endl; // 将输出s1=
cout << "s1.capacity()=" << s1.capacity() << endl; // 返回当前容量,可以存放字符的总数。
cout << "s1.size()=" << s1.size() << endl; // 返回容器中数据的大小。
cout << "容器动态数组的首地址=" << (void *)s1.c_str() << endl;
s1 = "xxxxxxxxxxxxxxxxxxxx";
cout << "s1.capacity()=" << s1.capacity() << endl; // 返回当前容量,可以存放字符的总数。
cout << "s1.size()=" << s1.size() << endl; // 返回容器中数据的大小。
cout << "容器动态数组的首地址=" << (void *)s1.c_str() << endl;
}
容器做了一次拓展,拓展后大小为字节
并且两个容器的首地址不同,string类拓展容器时,先分配更大的空间,然后把内容复制到新的空间,再把以前的空间释放掉。所以两个容器的地址肯定不一样。
2)string(const char *s); //使用C语言风格的字符串s初始化string对象。(转换构造函数)
例如:
int main()
{
string s2("hello world");
cout << "s2=" << s2 << endl; // 将输出s2=hello world
string s3 = "hello world";
cout << "s3=" << s3 << endl; // 将输出s3=hello world
}
3)string(const string &str); // 使用string对象str去初始化新创建的string对象(拷贝构造函数)。
int main()
{
string s3 = "hello world";
string s4(s3); //调用拷贝构造函数,s3 = "hello world";
cout << "s4=" << s4 << endl; // 将输出s4=hello world
string s5 = s3;//调用拷贝构造函数
cout << "s5=" << s5 << endl; // 将输出s5=hello world
}
s4和s5的内容都是从s3拷贝过来的。
注意string类中有一个指向动态数组的char*指针,所以string的拷贝构造函数一定是深拷贝。
string的拷贝构造函数中,前三个用的最多,后面4个,在实际开发,特别是处理文件和网络编程时,用的很多。
4)string(const char *s,size_t n); // 将string对象初始化为s指向的地址后n字节的内容。即使超过了C语言风格的字符串长度,只要小于n字节,后面的数据照样被复制。
🖲例如:
int main()
{
string s6("hello world", 5);//s6将指向h首字符后5字节
cout << "s6=" << s6 << endl; // 将输出s6=hello
string s7("hello world", 50);//s7将指向h首字符后50字节的内容,
cout << "s7=" << s7 << endl; //将输出s7=hello以及未知内容
}
对于字符串s7,如果遇到字符串的结尾/0,并不会停止复制。
5)string(const string &str,size_t pos=0,size_t n=npos); 从str的第pos位置开始,截取n个字符,用截取的内容创建string容器。
✅第一个参数是string对象,也可以是C语言风格的字符串,因为C++的string类内置了转换函数,可以将C语言的字符串风格给自动转换成string对象
✅第二个参数的缺省值是0,不是1。
✅第三个参数npos的缺省值是无限大。不指定npos就是截取pos位置后面的所有内容。
🖲例如:
int main()
{
string s3 = "hello world";
string s8(s3, 3, 5); // s3 = "hello world";
cout << "s8=" << s8 << endl; // 将输出s8=lo wo
string s9(s3, 3);//截取从s3字符串第3个位置的后的全部内容
cout << "s9=" << s9 << endl; // 将输出s9=lo world
cout << "s9.capacity()=" << s9.capacity() << endl; // 返回当前容量,可以存放字符的总数。
cout << "s9.size()=" << s9.size() << endl; // 返回容器中数据的大小。
cout << "s9.capacity()=" << s9.capacity() << endl; // 返回当前容量,可以存放字符的总数。
cout << "s9.size()=" << s9.size() << endl; // 返回容器中数据的大小。
//C++的string类内置了转换函数,可以将C语言的字符串风格给自动转换成string对象
string s10("hello world", 3, 5);
cout << "s10=" << s10 << endl; // 将输出s10=lo wo
}
注意这个构造函数和第四个不一样,它会判断str的结尾标志/0,然后停止,并不会无限的复制下去。上例对于s9字符串,从s3的第3个位置截取npos(无限大)个字符,并没有真正截取无限个字符,只截取了8个字符,遇到/0就停止了,所以s9容器的大小为8。
6)template string(T begin,T end); // 将string对象初始化为区间[begin,end]内的字符,其中begin和end的行为就像指针,用于指定位置,范围包括begin在内,但不包括end。
例如:
int main()
{
string s13 = "强风吹拂king";
string s14(s13.begin(), s13.end());
cout << "构造出来的字符串s14:" << s14 << endl;
return 0;
}
7)string(size_t n,char c); // 创建一个由n个字符c组成的string对象。
🖲例如:
int main()
{
string s12(8, 'x');
cout << "s12=" << s12 << endl; // 将输出s12=xxxxxxxx
cout << "s12.capacity()=" <<
s12.capacity() << endl; // s12.capacity()=15
cout << "s12.size()=" << s12.size() << endl; // s12.size()=8
string s13(30, 0);
cout << "s13=" << s13 << endl; // 将输出s13=
cout << "s13.capacity()=" <<
s13.capacity() << endl; // s13.capacity()=31
cout << "s13.size()=" << s13.size() << endl; // s12.size()=30
}
析构函数~string()释放内存空间。
二.string特性操作
💬特性函数表格一览:
函数原型 | 解释 |
---|---|
size_t max_size() const; | 返回string对象的最大长度string::npos,此函数意义不大。 |
size_t capacity() const; | 返回当前容量,可以存放字符的总数。 |
size_t length() const; | 返回容器中数据的大小(字符串语义)。 |
size_t size() const; | 返回容器中数据的大小(容器语义)。 |
bool empty() const; | 判断容器是否为空。 |
void clear(); | 清空容器,清空后,size()将返回0。 |
void shrink_to_fit(); | 将容器的容量降到实际大小(需要重新分配内存)。 |
void reserve( size_t size=0); | 将容器的容量设置为至少size。 |
void resize(size_t len,char c=0); | 把容器的实际大小置为len,如果len<实际大小,会截断多出的部分;如果len>实际大小,就用字符c填充。resize()后,length()和size()将返回len。 |
三.字符操作
💬表格一览:
函数原型 | 解释 |
---|---|
char& operator[](int n); | 通过[]方式取字符,通过[]访问元素,如果越界,不抛异常,程序直接挂掉 |
char& at(int n); | 通过at方法获取字符,如果越界,会抛异常 |
const char *c_str() const; | 返回容器中动态数组的首地址,语义:寻找以null结尾的字符串。 |
const char *data() const; | 返回容器中动态数组的首地址,语义:只关心容器中的数据。 |
int copy(char *s, int n, int pos = 0) const; | 把当前容器中的内容,从pos开始的n个字节拷贝到s中,返回实际拷贝的数目。 |
具体使用方法: |
#include <iostream>
#include<string>
using namespace std;
int main()
{
string s = "hello world";
for (int i = 0; i < s.size(); i++)
{
cout << s[i] << " ";
}
cout << endl;
for (int i = 0; i < s.size(); i++)
{
cout << s.at(i) << " ";
}
cout << endl;
}
关于[]和at异常的问题:
char& operator[](int n);函数
#include <iostream>
#include<string>
using namespace std;
int main()
{
string s = "hello world";
try
{
cout << s[100] << endl;
}
catch (out_of_range &ex)
{
cout << ex.what() << endl;
}
}
char& at(int n);函数
#include <iostream>
#include<string>
using namespace std;
int main()
{
string s = "hello world";
try
{
cout << s.at(100) << endl;
}
catch (out_of_range &ex)
{
cout << ex.what() << endl;
cout << "at越界" << endl;
}
}
四.赋值操作
给已存在的容器赋值,将覆盖容器中原有的内容。
💬表格一览:
函数原型 | 解释 |
---|---|
string& operator=(const char* s); | C语言风格字符串赋值给当前的string对象 |
string& operator=(const string&s); | 把string对象s赋给当前的string对象 |
string& operator=(char c); | 字符c赋值给当前的string对象 |
string& assign(const char *s); | 将C语言风格字符串s赋给当前的string对象 |
string &assign(const char *s,size_t n); | 把C语言风格字符串s的前n个字符赋给当前的string对象 |
string& assign(const string& str); | 把string对象str赋给当前的string对象 |
string& assign(size_t n,char c); | 用n个字符c赋给当前的string对象 |
string& assign(const string &str,size_t pos=0,size_t n=npos); | 将当前的string对象赋值为str从位置pos(缺省值为0,默认为首部)开始到n(缺省值npos,即默认为结尾) |
五.拼接操作
把内容追加到已存在容器的后面。
表格一览:
函数原型 | 解释 |
---|---|
string& operator+=(const string& str); | 重载+=操作符,将string对象str连接到当前string对象后。 |
string& operator+=(const char* s) | 重载+=操作符,将C语言风格字符串s连接到当前string对象后。 |
string& operator+=(const char c); | 重载+=操作符,将字符c连接到当前string对象后。 |
string& append(const char *s); | 把C语言风格字符串s连接到当前string对象结尾。 |
string &append(const char *s,size_t n); | 把C语言风格字符串s的前n个字符连接到当前string对象结尾。 |
string& append(const string&str); | 将string对象str连接到当前string对象后。 |
string& append(const string& s, int pos, int n); | 把string对象s中从pos开始的n个字符连接到当前string对象结尾。 |
string &append(size_t n,char c); | 在当前string容器结尾添加n个字符c。 |
六.交换操作
💬表格一览:
函数原型 | 解释 |
---|---|
void swap(string &str); | 把当前容器与str交换。 |
如果数据量很小,交换的是动态数组中的内容,如果数据量比较大,交换的是动态数组的地址。 |
七.string截取操作(子容器)
💬表格一览:
substr函数参数pos的缺省参数默认为0,即首部,参数n缺省参数默认为npos,即尾部。
函数原型(substr) | 解释 |
---|---|
string substr(size_t pos = 0,size_t n = npos) const; | 返回pos开始的n个字节组成的子容器。 |
八.string的比较操作
表格一览:
compare函数在>时返回 1,<时返回 -1,==时返回 0。比较区分大小写,比较时参考字典顺序,排越前面的越小。大写的A比小写的a小。
函数原型 | 解释 |
---|---|
bool operator==(const string &str1,const string &str2) const; | 比较两个字符串是否相等 |
int compare(const string& str) const; | 当前字符串与字符串str比较 |
int compare(size_t pos, size_t n,const string& str) const; | 比较当前字符串从pos开始的n个字符组成的字符串与str的大小 |
int compare(size_t pos, size_t n,const string &str,size_t pos2,size_t n2)const; | 比较当前字符串从pos开始的n个字符组成的字符串与str中pos2开始的n2个字符组成的字符串的大小 |
int compare(const char *s) const; | 当前字符串与C语言风格字符串s比较 |
int compare(size_t pos, size_t n,const char *s) const; | 比较当前字符串从pos开始的n个字符组成的字符串与C语言风格字符串s的大小 |
int compare(size_t pos, size_t n,const char *s, size_t pos2) const; | 比较当前字符串从pos开始的n个字符组成的字符串与C语言风格字符串s中从pos2位置开始的n个字符组成的字符串的大小 |
九.string的查找替换
💬find函数表格一览:
find函数参数pos的缺省值是0,即默认从头开始查找。
函数原型 (find) | 解释 |
---|---|
size_t find(const string& str, size_t pos = 0) const; | 查找str第一次出现位置,从pos位置开始查找。 |
size_t find(const char* s, size_t pos = 0) const; | 查找C语言风格字符串s第一次出现位置,从pos开始查找。 |
size_t find(const char* s, size_t pos, size_t n) const; | 从pos位置开始查找字符串s的前n个字符第一次出现的位置 |
size_t find(char c, size_t pos = 0) const; | 查找字符c第一次出现位置。 |
💬rfind函数表格一览:
rfind函数参数pos缺省参数为npos,即默认从尾部开始查找。
函数原型 | 解释 |
---|---|
size_t rfind(const string& str, size_t pos = npos) const; | 查找str第一次出现的位置,从pos位置开始查找 |
size_t rfind(const char* s, size_t pos = npos) const; | 查找C语言风格字符串s第一次出现位置,从pos位置开始查找 |
size_t rfind(const char* s, size_t pos, size_t n) const; | 从pos位置开始查找C语言风格字符串s的前n个字符第一次出现位置 |
int rfind(const char c, int pos = 0) const; | 查找字符c第一次出现的位置 |
十.string的替换
💬replace函数表格一览:
函数原型(replace) | 解释 |
---|---|
string& replace(size_t pos, size_t len, const string& str); | 从pos位置开始len长度的内容被替换为字符串str |
string& replace(size_t pos, size_t len, const char* s); | 从pos开始len长度的内容被替换为C语言风格字符串s。 |
string& replace(size_t pos, size_t len, const string& str, size_t subpos, size_t sublen = npos); | 从pos位置开始len长度的内容被替换为string对象str从位置subpos到sublen的内容。 |
string& replace(size_t pos, size_t len, const char* s, size_t n); | 从pos位置开始len长度的内容被替换为C语言风格字符串s前n个字符大小的内容。 |
string& replace(size_t pos, size_t len, size_t n, char c); | 从pos位置开始len长度的内容被n个字符c替换。 |
十一.string插入操作(insert)
💬insert函数一览:
函数原型 | 解释 |
---|---|
string& insert(size_t pos, const char* s); | 在pos位置插入C语言风格字符串s |
string& insert(size_t pos, const char* s, size_t n); | 在pos位置插入C语言风格字符串s前n个字符大小的内容。 |
string& insert(size_t pos, const string& str); | 在pos位置插入str |
string& insert(size_t pos, const string& str, size_t subpos, size_t sublen = npos); | 在pos位置插入string对象str从位置subpos到位置sublen大小的内容 |
string& insert(size_t pos, size_t n, char c); | 在pos位置插入n个字符c |
十二.string删除操作(erase)
💬erase函数一览:
函数原型 | 解释 |
---|---|
string& erase(int pos, int n = npos); | 删除从pos开始的n个字符 |