C++ string类(2)—成员访问、插入、删除、替换、查找和交换操作

目录

一、成员访问

1、[ ]&at

2、front( )&back( )

 二、插入元素

三、删除元素

四、替换元素

五、查找元素 

1、查找第一次出现位置

2 、在指定范围内查找

六、交换字符串

七、c_str

八、rfind&substr


一、成员访问

1、[ ]&at

虽然二者功能一样,但[ ]比较常用。

int main()
{string s1("hello world");cout << s1[4] << endl;cout << s1.at(4) << endl;return 0;
}

访问越界[ ]会直接报错,.at( )会抛异常。

int main()
{string s1("hello world");try {s1.at(100);}catch (const exception& e) {cout << e.what() << endl;}return 0;
}

2、front( )&back( )

int main()
{string s1("hello world");cout << s1.back() << endl;cout << s1.front() << endl;return 0;
}

 二、插入元素

insert/erase不推荐经常使用,能少用就少用,因为他们可能都存在要挪动数据,效率低下。

指定位置插入字符串。

int main()
{string s1("world");s1.insert(0, "hello");cout << s1 << endl;return 0;
}

指定位置插入单个字符 

s1.insert(0, "w");

 

 两种方式指定位置插入空字符

使用单引号需要加上插入字符个数(第二个参数),使用双引号不需要。

int main()
{string s1("world");s1.insert(2, 1, ' ');s1.insert(3, " ");cout << s1 << endl;return 0;
}

 

 插入可以借助迭代器的begin()和end()获取位置。

s1.insert(s1.begin()+2, ' ');
s1.insert(s1.end() - 1, ' ');

 

三、删除元素

 insert/erase不推荐经常使用,能少用就少用,因为他们可能都存在要挪动数据,效率低下。 

删除指定位置指定长度字符。 

int main()
{string s2("hello world");s2.erase(5, 1);cout << s2 << endl;return 0;
}

也可以使用迭代器。 

    s2.erase(s2.begin() + 5);

 

 如果长度大于字符串长度或者缺省,则删除指定位置开始一直到最后的字符。

	s2.erase(5, 30);s2.erase(5);

 

四、替换元素

replace()函数的第一个参数是替换的起始位置,第二个参数是要替换的字符数,第三个参数是替换的字符串。 

int main()
{string s1("hello world");s1.replace(5, 1, "&&&&");cout << s1 << endl;return 0;
}
  • 首先,代码中的第一行是被注释掉的,表示字符串s1被初始化为"hello world"。这是一个注释行,不会被执行。
  • 然后,使用s1.replace(5, 1, "&&&&")函数调用来替换字符串s1中从索引位置5开始的1个字符,将其替换为"&&&&"。
  • 接下来,使用cout对象和<<运算符将修改后的字符串s1输出到标准输出流。

使用replace时,空间不够扩容,还要移动数据。 

五、查找元素 

1、查找第一次出现位置

find()用于在字符串中查找指定子串的第一个出现位置。

find()函数有多个重载版本,其中最常用的版本接受一个参数,即要查找的子串。它返回一个整数值,表示子串在字符串中的位置索引。如果找到了子串,则返回第一个匹配的位置索引;如果未找到子串,则返回一个特殊的值std::string::npos

int main()
{string s1("hello world I love you");size_t pos = s1.find(' ');while (pos != string::npos) {s1.replace(pos, 1, "***");pos = s1.find(' ');}cout << s1 << endl;return 0;
}

我们可以对上述程序进行优化。

int main()
{string s1("hello world I love you");size_t num = 0;for (auto ch : s1) {if (ch == ' ')++num;}s1.reserve(s1.size() + 2 * num);size_t pos = s1.find(' ');while (pos != string::npos) {s1.replace(pos, 1, "***");pos = s1.find(' ',pos+3);}cout << s1 << endl;return 0;
}

首先,在代码中创建了一个名为s1的字符串对象,并将其初始化为"hello world I love you"。

然后,使用一个循环遍历字符串s1中的每个字符。在循环中,如果当前字符是空格,则将计数器num加1。

接下来,使用s1.reserve(s1.size() + 2 * num)函数调整字符串s1的容量,以便能够容纳替换后的字符串。这样做是为了避免在替换过程中频繁地重新分配内存,提高性能

然后,使用s1.find(' ', pos)函数来查找字符串s1中下一个空格的位置。如果找到了空格,将其位置存储在变量pos中。

接下来,使用s1.replace(pos, 1, "***")函数将找到的空格替换为三个星号"***"。这样做会修改字符串s1中的内容。

然后,使用pos = s1.find(' ', pos + 3)来查找下一个空格的位置,从上一个空格的位置加3开始查找。这样做是为了避免重复替换已经被替换过的空格

循环会一直执行,直到没有更多的空格被找到。

最后,使用cout对象和<<运算符将修改后的字符串s1输出到标准输出流。

 还可以使用+=运算符进一步优化。

int main()
{string s1("hello world I love you");string newStr;size_t num = 0;for (auto ch : s1) {if (ch != ' ')++num;}newStr.reserve(s1.size() + 2 * num);for (auto ch : s1) {if (ch != ' ')newStr += ch;elsenewStr += "***";}s1 = newStr;
}

与之前的版本相比,这段程序是对第一段程序的改进版本,具有以下特点和优点: 

  1. 更简洁:第二段程序使用了更简洁的方法来替换字符串中的空格,避免了使用循环和查找函数。

  2. 更高效:第二段程序只需遍历一次原始字符串,而不是使用循环和查找函数多次遍历。这样可以减少时间复杂度,提高程序的执行效率。

  3. 更易读:第二段程序使用了更直观的方式来替换空格,通过判断字符是否为空格来决定添加字符还是添加"***"。这样代码更易读懂,减少了冗余的操作。

  4. 更节省内存:第二段程序使用了新的字符串newStr来存储替换后的结果,避免了对原始字符串s1进行频繁的修改。这样可以减少内存的使用,提高程序的效率。

2 、在指定范围内查找

六、交换字符串

int main()
{string s1("hello world");string s2("xxxxx");s1.swap(s2);cout << s1 << endl;cout << s2 << endl;swap(s1, s2);cout << s1 << endl;cout << s2 << endl;
}

这段程序中的两个swap函数调用有以下区别:

  1. s1.swap(s2):这是成员函数形式的swap调用,它将s1s2两个字符串对象进行交换。在交换后,s1的值变为"xxxxx",s2的值变为"hello world"。这种形式的swap函数是通过成员函数调用来实现的。

  2. swap(s1, s2):这是非成员函数形式的swap调用,它通过传递s1s2两个字符串对象作为参数来进行交换。在交换后,s1的值变为"hello world",s2的值变为"xxxxx"。这种形式的swap函数是通过非成员函数调用来实现的。

总结起来,这两个swap函数的功能是一样的,都是用于交换两个字符串对象的值。区别在于调用方式不同,一个是通过成员函数调用,一个是通过非成员函数调用。使用哪种形式的swap函数取决于个人偏好和代码风格。

 

七、c_str

int main()
{string s1("hello world");cout << s1 << endl;cout << s1.c_str() << endl;return 0;
}
  1. cout << s1 << endl;:这行代码将字符串对象s1直接输出到标准输出流cout中。它会输出s1的内容,即字符串 "hello world"。

  2. cout << s1.c_str() << endl;:这行代码使用了字符串对象s1c_str()成员函数。c_str()函数返回一个指向以空字符结尾的字符数组(C风格字符串)的指针。然后,该指针被传递给cout输出流进行输出。它也会输出s1的内容,即字符串 "hello world"。

区别在于输出的方式不同。第一行直接输出字符串对象s1的内容,而第二行使用了c_str()函数将字符串对象转换为C风格字符串后输出。通常情况下,直接输出字符串对象更为简洁和方便。

int main()
{string s1("hello world");cout << s1 << endl;cout << s1.c_str() << endl;s1 += '\0';s1 += '\0';s1 += "xxxxx";cout << s1 << endl;cout << s1.c_str() << endl;return 0;
}

可以看到,通过对s1进行修改操作后,输出结果中的空字符('\0')在字符串对象s1的输出中仍然存在,但在c_str()函数返回的C风格字符串中被忽略了。这是因为c_str()函数会将字符串以空字符结尾,而在输出时遇到空字符就会停止输出。 

我们逐步解释为什么输出结果会有所不同。

初始状态下,s1的值为"hello world"。

  1. cout << s1 << endl;:输出s1的内容,即字符串 "hello world"。

  2. cout << s1.c_str() << endl;:输出s1的C风格字符串表示,即 "hello world"。

  3. s1 += '\0';:在s1末尾添加一个空字符('\0'),此时s1的值变为 "hello world\0"。

  4. s1 += '\0';:再次在s1末尾添加一个空字符('\0'),此时s1的值变为 "hello world\0\0"。

  5. s1 += "xxxxx";:将字符串 "xxxxx" 追加到s1的末尾,此时s1的值变为 "hello world\0\0xxxxx"。

  6. cout << s1 << endl;:输出s1的内容,即 "hello world\0\0xxxxx"。

  7. cout << s1.c_str() << endl;:输出s1的C风格字符串表示,即 "hello world"。这里需要注意,c_str()函数返回的是以空字符结尾的字符数组,输出时遇到第一个空字符就会停止输出。

八、rfind&substr

rfind函数用于在一个字符串中从后往前搜索指定的子字符串,并返回子字符串的位置。

它的语法如下:

size_t rfind(const string& str, size_t pos = string::npos) const;

str是要搜索的子字符串,pos是搜索的起始位置,默认为string::npos,表示从字符串的末尾开始搜索。

substr函数用于从一个字符串中提取子字符串。它的语法如下:

string substr(size_t pos = 0, size_t len = string::npos) const;

其中,pos是要提取的子字符串的起始位置,默认为0,len是要提取的子字符串的长度,默认为string::npos,表示提取从起始位置到字符串末尾的所有字符。

接下来看下面代码:

int main()
{string file("string.cpp.tar.zip");size_t pos = file.rfind('.');if (pos != string::npos){string suffix = file.substr(pos);cout << suffix << endl;}return 0;
}

 

这段代码的目的是提取文件名中的后缀名。

首先,定义了一个字符串file,其中包含了一个文件名string.cpp.tar.zip

然后,使用rfind函数从后往前搜索.字符的位置,并将结果保存在变量pos中。如果找到了.字符,则pos的值不等于string::npos

接下来,通过判断pos的值是否不等于string::npos,来确定是否找到了.字符。如果找到了,则使用substr函数从pos位置开始提取子字符串,并将结果保存在变量suffix中。

最后,将提取到的后缀名输出到标准输出流cout中,然后换行。

在这个例子中,输出结果为.zip,因为.字符后面的部分就是文件的后缀名。

 从URL中提取主机地址:

int main()
{string url("http://www.cplusplus.com/reference/string/string/find/");cout << url << endl;size_t start = url.find("://");if (start == string::npos){cout << "invalid url" << endl;}start += 3;size_t finish = url.find('/', start);string address = url.substr(start, finish - start);cout << address << endl;return 0;
}

使用find函数搜索字符串url中第一次出现的子字符串"://"的位置,并将结果保存在变量start中。如果找不到该子字符串,则start的值等于string::npos

接下来,通过判断start的值是否等于string::npos,来确定是否找到了"://"子字符串。如果没有找到,则输出"invalid url",表示URL无效。

如果找到了"://"子字符串,则将start的值增加3,以跳过"://"部分,然后使用find函数搜索从start位置开始的下一个'/'字符的位置,并将结果保存在变量finish中。

最后,使用substr函数从start位置开始,提取从startfinish之间的子字符串,并将结果保存在变量address中,将提取到的主机地址输出到标准输出流cout中,然后换行。

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

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

相关文章

数据结构奇妙旅程之顺序表和链表

꒰˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN …

C++-内联函数

目录 一.什么是内联函数 1.内联函数的概念 2.内联函数的定义 二.C中引入内联函数的原因 三.什么样的函数适合被声明为内联呢&#xff1f; 四.面试题 一.什么是内联函数 1.内联函数的概念 以inline修饰的函数叫做内联函数&#xff0c;编译时C编译器会在调用内联函数的地方展开…

springboot 2.4.4集成 hikari连接池多数据源实例

文章目录 前言一、配置步骤1.1 pom配置1.2 application.properties配置1.3 DataSourceContextHolder类1.4 DynamicDataSource1.5 DataSourceconfig类配置1.6 配置TargetDataSource注解1.7 切面方法1.8 dao的写法 二、测试验证2.1 启动springboot项目2.2 检查数据库连接2.3 debu…

python——进程常用功能

Python的multiprocessing模块提供了强大的并行处理能力&#xff0c;以下是几个功能的详细解释&#xff1a; join(): 在multiprocessing中&#xff0c;join方法用于阻塞主进程直到指定的进程终止。这对于确保所有子进程在程序结束前完成其工作是很有用的。deamon(): 在multipro…

基于51单片机的十字路口交通灯_5s黄灯倒计时闪烁

基于51单片机十字路口交通灯_5s黄灯闪烁 &#xff08;程序仿真仿真视频&#xff09; 仿真&#xff1a;proteus 7.8 程序编译器&#xff1a;keil 4/keil 5 编程语言&#xff1a;C语言 设计编号&#xff1a;J006 功能要求 交通灯运行状态&#xff1a; &#xff08;1&…

硬件基础:二极管

基本定义 二极管的内部其实就是一个PN结。 把PN结封装起来&#xff0c;两边加上两个电极&#xff0c;就组成了半导体二极管。简称二极管&#xff08;Diode&#xff09; 二极管和PN结一样&#xff0c;具有单向导通性&#xff1a; 外观和正负极 常见芯片封装如下&#xff1a; 一般…

java 工具类: CompareUtils(比较对象字段值变化)

一、前言 我们在工作中&#xff0c;可能会在日志中记录数据的变化情况或者在公共处理的数据增加一个日志页面&#xff0c;记录每次修改的变化。我们可以根据CompareUtils工具类比较数据前后发生了怎样的变化, 这样我们就可以知道数据做了哪些改变. 二、条件限制 在写这个通用…

ImportError: cannot import name ‘metadata‘ from ‘importlib‘

yolov8 编译问题 ImportError: cannot import name ‘metadata’ from ‘importlib’ 将 from importlib import metadata 更改为 import importlib_metadata as metadata

【WinForm.NET开发】创建 Windows 窗体应用

本文内容 创建项目创建应用程序运行应用程序 本文演示创建一个具有基于 Windows 的用户界面 (UI) 的简单 C# 应用程序。 1、创建项目 首先&#xff0c;创建 C# 应用程序项目。 项目类型随附了所需的全部模板文件&#xff0c;无需添加任何内容。 打开 Visual Studio。在“开…

vs 安装 qt qt扩展

1 安装qt 社区版 免费 Download Qt OSS: Get Qt Online Installer 2 vs安装 qt vs tools 3 vs添加 qt添加 bin/cmake.exe 路径 3.1 扩展 -> qt versions 3.2

【设计模式-4.1】行为型——观察者模式

说明&#xff1a;本文介绍设计模式中行为型设计模式中的&#xff0c;观察者模式&#xff1b; 商家与顾客 观察者模式属于行为型设计模式&#xff0c;关注对象的行为。以商家与顾客为例&#xff0c;商家有商品&#xff0c;顾客来购买商品&#xff0c;如果商家商品卖完了&#…

Vue+ElementUI+C#前后端分离:监控长耗时任务的实践

想象一下&#xff0c;我们正在构建一个Web应用&#xff0c;需要实现一个数据报告的导出功能。这听起来很简单&#xff0c;不是吗&#xff1f;但是&#xff0c;随着深入开发&#xff0c;我们意识到导出过程比预期的要复杂和耗时得多。由于报告的数据量巨大&#xff0c;后端需要花…

PostgreSQL有意思的现象:支持不带列的表

1、前言 以前从没有试过建一张表&#xff0c;不带任何列。在PG中却支持这种语法。这是个什么鬼? 最近&#xff0c;把PG源码扒了下&#xff0c;简单浏览了下最近的一些merge。其中有一个fix&#xff1a; eeb0ebad79 ("Fix the initial sync tables with no columns.&qu…

〖大前端 - 基础入门三大核心之JS篇㊺〗- 定时器和延时器

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…

3D场景建模工具

在线工具推荐&#xff1a; 三维数字孪生场景工具 - GLTF/GLB在线编辑器 - Three.js AI自动纹理化开发 - YOLO 虚幻合成数据生成器 - 3D模型在线转换 - 3D模型预览图生成服务 1. 什么是3D场景建模&#xff1f; 3D场景建模是一种通过计算机图形学技术&#xff0c;将现实世…

【Matlab】如何快速入门一项新技能-以Matlab/Simulink入门为例

目录 1. 引言 2. 背景 3. 快速学习并完成开发 3.1 了解需求&#xff0c;知道要干什么 3.2 了解Matlab/Simulink基本功能 第一步&#xff0c;查看Matlab的中文网站中文网站https://www.ilovematlab.cn/resources/对Matlab/Simulink有了一个初步认识。 3.3 实现一个最简单…

PyQt6 QDialogButtonBox组合按钮控件

锋哥原创的PyQt6视频教程&#xff1a; 2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计34条视频&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面开发 视频教程(无废话版…

【开源威胁情报挖掘1】引言 + 开源威胁情报挖掘框架 + 开源威胁情报采集与识别提取

基于开源信息平台的威胁情报挖掘综述 写在最前面摘要1 引言近年来的一些新型网络安全威胁类型挖掘网络威胁的情报信息威胁情报分类&#xff1a;内、外部威胁情报国内外开源威胁情报挖掘分析工作主要贡献研究范围和方法 2 开源威胁情报挖掘框架1. 开源威胁情报采集与识别2. 开源…

软件生命周期四个阶段SDLC

软件产品生命周期&#xff1a;指软件产品研发全部过程、活动和任务的结构框架。 产品的生命周期一般包括四个阶段&#xff1a;引入期、成长期、成熟期和衰退期&#xff0c;在不同的阶段中&#xff0c;市场对产品的反应不同&#xff0c;其销售特点不同&#xff0c;因而产品管理的…

mysql数据库的配置文件在哪里

可以搜索my.ini、或者my.cnf&#xff0c;看看在哪个地方。 例如&#xff0c;我在windows系统装的mysql 8.2版本&#xff0c;my.ini文件不在安装目录下&#xff0c;而在另外一个目录下。 我的安装目录是F:\Program Files\MySQL\MySQL Server 8.2&#xff0c;但my.ini文件在C:\Pr…