C++——string容器常用操作汇总

在这里插入图片描述
纵有疾风起,人生不言弃。本文篇幅较长,如有错误请不吝赐教,感谢支持。

💬文章目录

    • 一.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 worldstring 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 worldstring 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=hellostring 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 wostring s9(s3, 3);//截取从s3字符串第3个位置的后的全部内容cout << "s9=" << s9 << endl; // 将输出s9=lo worldcout << "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=xxxxxxxxcout << "s12.capacity()=" <<s12.capacity() << endl; // s12.capacity()=15cout << "s12.size()=" << s12.size() << endl; // s12.size()=8string s13(30, 0);
cout << "s13=" << s13 << endl; // 将输出s13=cout << "s13.capacity()=" <<s13.capacity() << endl; // s13.capacity()=31cout << "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个字符

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/3218.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

亚马逊云科技积极探索多样化生态合作模式,助力实现可持续发展愿景

2023年6月26日&#xff0c;亚马逊云科技中国峰会在上海世博中心盛大开幕&#xff01;以主题“因构建 而可见”为大家拉开帷幕。当前&#xff0c;越来越多的企业客户&#xff0c;以及当地政府监管机构对企业的要求&#xff0c;都需要企业告知碳足迹&#xff0c;亚马逊云科技提供…

华为OD机试真题 JavaScript 实现【寻找峰值】【牛客练习题】

一、题目描述 给定一个长度为n的数组nums&#xff0c;请你找到峰值并返回其索引。数组可能包含多个峰值&#xff0c;在这种情况下&#xff0c;返回任何一个所在位置即可。 1.峰值元素是指其值严格大于左右相邻值的元素。严格大于即不能有等于&#xff1b; 2.假设 nums[-1] n…

C语言:数据的存储

往期文章 C语言&#xff1a;初识C语言C语言&#xff1a;分支语句和循环语句C语言&#xff1a;函数C语言&#xff1a;数组C语言&#xff1a;操作符详解C语言&#xff1a;指针详解C语言&#xff1a;结构体 目录 往期文章前言1. 数据的类型2. 整型在内存中的存储2.1 原码、反码、…

不知道识别表格的方式有哪些?分享识别表格怎么弄

小明&#xff1a;嘿&#xff0c;小红&#xff01;你知道吗&#xff1f;最近我在整理一堆纸质表格&#xff0c;但是手动输入数据实在太耗时间了&#xff0c;我在想有没有什么方法可以快速识别表格的内容呢&#xff1f; 小红&#xff1a;哦&#xff0c;我听说过有一些方式可以自…

认识固态继电器及其工作原理

什么是固态继电器&#xff0c;有什么优缺点&#xff1f; 固态继电器 简称SSR&#xff0c;又被称之为“无触点开关”它利用电子元件&#xff08;如双向可控硅等半导体器件&#xff09;的开关特性&#xff0c;可到达无触点无火花地接通和断开电路。 固态继电器工作可靠&#…

大数据开发之Hive案例篇14:某个节点HDFS块比较多

文章目录 一. 问题描述二. 解决方案2.1 查看节点安装的组件2.2 排查HDFS配置2.3 排查Yarn配置2.3.1 首先查看下nodemanager的日志2.3.2 查看container分配情况2.3.3 查看调度机制2.3.4 查看集群任务情况2.3.5 集群负载情况2.3.6 resourcemanager与nodemanager是否可以混合部署 …

【ArcGIS Pro二次开发】(44):属性结构描述表【Excel】转空库(批量)

随着县级国土空间总体规划数据库规范的下发&#xff0c;建立标准空库是一项马上就要着手的工作。国空的数据库体量很大&#xff0c;单是要素类就有100多个&#xff0c;不是以前村规数据库能比的&#xff0c;手动建库是不可能的&#xff0c;工具自动建库就是一个很合理的选择。 …

JavaScript数据存储方式

内置对象 js内部提供的对象&#xff0c;包含各种属性和方法给开发者调用 document.write() console.log() Math Math对象是js提供的一个 “数学”对象&#xff0c;提供了一系列做数学运算的方法 max找最大值Math.max(3,8,5,4) 返回8min找最小值Math.min(3,8,5,4) 返回4ab…

nginx部署本地项目如何让异地公网访问?服务器端口映射配置!

接触过IIS或apache的小伙伴们&#xff0c;对nginx是比较容易理解的&#xff0c;nginx有点类似&#xff0c;又有所差异&#xff0c;在选择使用时根据自己本地应用场景来部署使用即可。通过一些对比可能会更加清楚了解&#xff1a; 1.nginx是轻量级&#xff0c;比apache占用更少…

【Matlab】神经网络遗传算法函数极值寻优——非线性函数求极值

目前关于神经网络遗传算法函数极值寻优——非线性函数求极值的博客资源已经不少了&#xff0c;我看了下来源&#xff0c;最初的应该是来自于Matlab中文论坛&#xff0c;论坛出版的《MATLAB神经网络30个案例分析》第4章就是《神经网络遗传算法函数极值寻优——非线性函数极值寻优…

深入理解浏览器的缓存机制之协商缓存与强缓存

目录 什么是浏览器缓存 浏览器缓存的分类 &#x1f397;️ 强缓存 &#x1f397;️ 协商缓存 &#x1f449;&#x1f3fb; 缓存请求流程 &#x1f449;&#x1f3fb; 为什么要有Etag &#x1f449;&#x1f3fb; 缓存优先级 &#x1f449;&#x1f3fb; 启发式缓存 &a…

DBeaver连接mysql时报错com.mysql.cj.jdbc.Driver的解决方法【修改驱动下载的maven地址和重新下载驱动】

文章目录 说明解决方法1、打开DBeaver点击窗口-->窗口-->首选项-->链接-->点击驱动-->Maven-->添加2、删除已有的驱动3、重新创建mysql链接 说明 网上下载了最新版本的DBeaver软件&#xff0c;但是链接mysql的时候驱动下载失败&#xff0c;所以就报下面错误…