C++深入学习string类成员函数(1):默认与迭代
引言
在 C++ 编程中,std::string 类是处理字符串的核心工具之一。作为一个动态管理字符数组的类,它不仅提供了丰富的功能,还通过高效的内存管理和操作接口,极大地方便了字符串操作。通过深入探讨 std::string 的各类成员函数,我们将更加全面地理解它背后的机制和使用场景。
在本系列博客中,我们将详细剖析 std::string 的各类成员函数,带你深入理解 C++ 中字符串的操作与管理。以下内容均是在c++11标准版本的介绍。
1.默认成员函数
1.1string类的构造函数(Construct)
1.1.1函数原型
在C++中,std::string 是标准库提供的一个字符串类,它提供了多种构造函数来创建字符串对象。构造基本语法如下:
public member function
std::string::stringdefault (1) string();copy (2) string (const string& str);from c-string (3) string (const char* s);from sequence (4) string (const char* s, size_t n);fill (5) string (size_t n, char c);substring (6) string (const string& str, size_t pos, size_t len = npos);range (7) template <class InputIterator>string (InputIterator first, InputIterator last);initializer list (8) string (initializer_list<char> il);move (9) string (string&& str) noexcept;
1.1.2 noexcept
noexcept是c++11引入的关键字,用来表明不会抛出异常,它在函数声明中使用,告诉编译器和程序员该函数在运行时不会引发异常。
(1)noexcept 的作用:
优化性能:编译器可以利用 noexcept 进行优化。如果编译器知道某个函数不会抛出异常,它可以避免生成一些与异常处理相关的代码,从而提高性能。
提高代码安全性:当你明确声明某个函数不会抛出异常,调用者可以放心地使用该函数,而不需要担心异常的传播和处理。
异常安全保障:当你编写不应该抛出异常的函数时,使用 noexcept 明确表达这种意图,防止由于代码错误导致异常意外抛出。
(2)使用场景
移动构造函数和移动赋值运算符:对于某些类型的对象,移动操作(如移动构造函数、移动赋值运算符)通常是 noexcept 的。这是因为在某些容器(如 std::vector)中,出于性能考虑,当元素在容器内部移动时,要求这些操作不会抛出异常。
异常安全的保证:如果你的函数不打算抛出异常,最好明确使用 noexcept 进行标记,提升代码的可读性和可维护性。
1.1.3深入解析
(1) 默认构造函数
创建一个空字符串对象。
std::string str;
此时,`str` 是一个空字符串。
(2)拷贝构造函数
通过另一个 `std::string` 对象来构造字符串。
string (const string& str);std::string str1 = "Hello";
std::string str2 = str1; // 使用str1来构造str2
(3)从 C 字符串构造
使用 C 风格的字符串(即字符数组)来构造 `std::string` 对象。
string (const char* s);const char* cstr = "Hello";
std::string str(cstr);
(4)从 C 字符串的部分字符构造
可以指定从 C 字符串中提取的字符个数。
string (const char* s, size_t n);const char* cstr = "Hello, World!";
std::string str(cstr, 5); // 只提取前5个字符,即"Hello"
(5)从字符构造
可以指定使用某个字符构造一个指定长度的字符串。
string (size_t n, char c);std::string str(5, 'A'); // 构造一个包含5个'A'字符的字符串 "AAAAA"
(6)从子字符串开始构造
可以从现有的std::string对象中提取一个子字符串来构造新的std::string。pos是子字符串的起始位置,len表示要提取的字符数(默认为npos,表示从起始位置到字符串末尾)。
string (const string& str, size_t pos, size_t len = npos); std::string original = "Hello, World!";// 从位置7开始,提取5个字符构造子字符串std::string subStr(original, 7, 5);// 输出 "World"
(7)从迭代器范围构造
可以使用迭代器范围来构造字符串,例如从数组或其他容器中提取内容。
template <class InputIterator>string (InputIterator first, InputIterator last);std::vector<char> vec = {'H', 'e', 'l', 'l', 'o'};
std::string str(vec.begin(), vec.end()); // 构造 "Hello"
(8)初始化构造列表(C++11及更高版本)
std::string 的 initializer list 构造函数允许你使用一个 初始化列表(std::initializer_list<char>)直接初始化一个字符串对象。
string (initializer_list<char> il);std::string str = {'H', 'e', 'l', 'l', 'o'};std::cout << "str: " << str << std::endl; // 输出 "Hello"
(9)移动构造函数 (C++11及更高版本)
可以利用移动语义,从一个临时的 `std::string` 对象中转移资源,而不需要复制。
string (initializer_list<char> il);std::string str1 = "Hello";
std::string str2 = std::move(str1); // str2获得str1的资源,str1被置为空
总结
`std::string` 的构造方式灵活多样,可以从字符数组、部分字符、字符迭代器等多种来源构造。此外,在C++11及以上版本中,`std::string` 还支持移动构造,进一步优化了性能。
1.2赋值运算符重载(operator=)
1.2.1函数原型:
string (1) string& operator= (const string& str);c-string (2) string& operator= (const char* s);
character (3) string& operator= (char c);
initializer list (4) string& operator= (initializer_list<char> il);move (5) string& operator= (string&& str) noexcept;
1.2.2深入解析
(1)拷贝赋值运算符
拷贝赋值运算符用于将一个 `std::string` 对象赋值给另一个 `std::string` 对象。
string& operator= (const string& str);std::string str1 = "Hello";std::string str2;str2 = str1; // 使用拷贝赋值运算符std::cout << "str1: " << str1 << std::endl; // 输出 "Hello"std::cout << "str2: " << str2 << std::endl; // 输出 "Hello"
(2)C 风格字符串赋值运算符
用于将从风格的字符串(字符数组指针)赋值给std::string对象
string& operator= (const char* s);const char* cStr = "Hello, World!";std::string str;str = cStr; // 使用C风格字符串赋值运算符std::cout << "str: " << str << std::endl; // 输出 "Hello, World!"
(3)字符赋值运算符
用于将一个字符赋值给 `std::string` 对象,将该字符作为字符串的唯一字符。
string& operator= (char c);char c = 'A';std::string str;str = c; // 使用字符赋值运算符std::cout << "str: " << str << std::endl; // 输出 "A"
(4)初始化列表赋值运算符
用于通过 C++11 提供的初始化列表,将多个字符赋值给 `std::string` 对象。
string& operator= (std::initializer_list<char> il);std::string str;str = {'H', 'e', 'l', 'l', 'o'}; // 使用初始化列表赋值运算符std::cout << "str: " << str << std::endl; // 输出 "Hello"
(5)移动赋值运算符
用于将一个临时或即将被销毁的 `std::string` 对象的内容移动到当前对象,而不是复制。
string& operator= (string&& str) noexcept;string str1 = "hello";string str2;str2 = std::move(str1);cout << str1 << endl; // str1 可能为空或处于未定义状态cout << str2 << endl;// 输出 "Hello"
1.2.3总结
- `string& operator=(const string& str)`:将一个 `std::string` 对象赋值给另一个。
- `string& operator=(const char* s)`:将 C 风格字符串赋值给 `std::string`。
- `string& operator=(char c)`:将一个字符赋值给 `std::string`,该字符成为字符串的唯一字符。
- `string& operator=(initializer_list<char> il)`:使用初始化列表赋值多个字符。
- `string& operator=(string&& str) noexcept`:使用移动赋值运算符,将一个临时对象的内容移动到当前对象。
这些赋值运算符提供了不同的赋值方式,极大地增强了 `std::string` 的灵活性。
2.返回迭代器的成员函数
在 C++中,迭代器是一种对象,它能够用来遍历容器中的元素。迭代器提供了一种统一的方式来访问不同类型容器中的元素,而无需了解容器的内部实现细节。这里如果对迭代器不太了解,可以去看这篇博客C++标准库类——string类-CSDN博客。
std::string 提供了一系列返回迭代器的成员函数,用于遍历和操作字符串中的字符。通过这些迭代器,您可以轻松地访问、修改和遍历字符串中的每个字符。常见的迭代器包括正向迭代器、常量迭代器、反向迭代器以及它们的常量版本。
2.1正向迭代器(iterator)
(1)begin()
- begin():返回指向字符串首个字符的迭代器。
iterator begin() noexcept;
const_iterator begin() const noexcept;
- iterator begin() noexcept 返回一个可修改的迭代器,指向字符串的第一个字符。
- const_iterator begin() const noexcept 返回一个常量迭代器,指向字符串的第一个字符,且不允许通过该迭代器修改字符串内容。
(2)end()
- end():返回指向字符串末尾(即最后一个字符的下一个位置)的迭代器。
iterator end() noexcept;
const_iterator end() const noexcept;
- iterator end()返回一个可修改的迭代器,指向字符串末尾的下一个位置。
- const_iterator end() const noexcept返回一个常量迭代器,指向末尾的下一个位置,且不允许通过该迭代器修改字符串内容。
(3)示例
#include <iostream>
using namespace std;int main()
{//使用非const迭代器遍历并修改字符string st1 = "Hello, henu";string::iterator it1 = st1.begin();while (it1 != st1.end()){if (*it1 == ',')*it1 = ';';it1++;}cout << st1 << endl;//Hello; henu//使用const迭代器遍历,但不能修改字符const string st2 = "Hello, world";string::const_iterator it2 = st2.begin();while (it2 != st2.end()){cout << *it2;//Hello, worldit2++;}return 0;
}
(4)常量版本
- cbegin():返回指向字符串首个字符的常量迭代器。
const_iterator cbegin() const noexcept;
- cend():返回指向字符串末尾的常量迭代器。
const_iterator cend() const noexcept;
常量版本其实就是与上面返回常量迭代器的声明类型的作用是相同的,在此不再赘述。
2. 2反向迭代器(reverse_iterator)
(1)rbegin()
- rbegin():返回指向字符串末尾(即最后一个字符)的反向迭代器。
reverse_iterator rbegin() noexcept;
const_reverse_iterator rbegin() const noexcept;
- reverse_iterator rbegin() 返回一个可修改的反向迭代器,指向字符串的最后一个字符。
- const_reverse_iterator rbegin() const noexcept 返回一个常量反向迭代器,指向字符串的最后一个字符,且不允许通过该迭代器修改字符串内容。
(2)rend()
- rend():返回指向字符串首个字符的前一个位置的反向迭代器。
reverse_iterator rend() noexcept;
const_reverse_iterator rend() const noexcept;
- reverse_iterator rend() 返回一个可修改的反向迭代器,指向第一个元素之前的位置。
- const_reverse_iterator rend() const noexcept 返回一个常量反向迭代器,指向第一个元素之前的位置,且不允许通过该迭代器修改字符串内容。
(3)示例
#include <iostream>
using namespace std;int main()
{//使用 reverse_iterator遍历并修改字符string st1 = "Hello, world";//auto it1 = st1.rbegin();string::reverse_iterator it1 = st1.rbegin();while (it1 != st1.rend()){if (*it1 == ',')*it1 = ';';cout << *it1;it1++;//注意,这里用++,而不是--}cout << endl;//使用 const_reverse_iterator遍历并修改字符const string st2 = "Hello, henu";//auto it2 = st2.rbegin();string::const_reverse_iterator it2 = st2.rbegin();//string::const_reverse_iterator it2 = st2.crbegin();while (it2 != st2.rend())//while (it2 != st2.crend()){cout << *it2;it2++;}return 0;
}

(4)常量版本:
- crbegin():返回指向字符串末尾的常量反向迭代器。
const_reverse_iterator crbegin() const noexcept
- crend():返回指向字符串首个字符的前一个位置的常量反向迭代器。
const_reverse_iterator crend() const noexcept;
这里的常量版本同样与上面返回常量迭代器的声明的类型的作用相同,不再赘述。
2.3总结
- begin():正向迭代器,指向首个字符。
- end():正向迭代器,指向末尾后的位置。
- rbegin():反向迭代器,指向最后一个字符。
- rend():反向迭代器,指向第一个字符之前的位置。
- cbegin():常量正向迭代器,指向首个字符。
- cend():常量正向迭代器,指向末尾后的位置。
- crbegin():常量反向迭代器,指向最后一个字符。
- crend():常量反向迭代器,指向第一个字符之前的位置。
更多string类的成员函数:string - C++ Reference (cplusplus.com)
相关文章:
C++深入学习string类成员函数(1):默认与迭代
引言 在 C 编程中,std::string 类是处理字符串的核心工具之一。作为一个动态管理字符数组的类,它不仅提供了丰富的功能,还通过高效的内存管理和操作接口,极大地方便了字符串操作。通过深入探讨 std::string 的各类成员函数&#…...
DataGrip远程连接Hive
学会用datagrip远程操作hive 连接前提条件: 注意:mysql是否是开启状态 启动hadoop集群 start-all.sh 1、启动hiveserver2服务 nohup hiveserver2 >> /usr/local/soft/hive-3.1.3/hiveserver2.log 2>&1 & 2、beeline连接 beelin…...
go 读取excel
一、安装依赖 go get github.com/tealeg/xlsx二、main.go package mainimport "fmt" import "github.com/tealeg/xlsx"type Student struct {Name stringSex string }func (student Student) show() {fmt.Printf("Name:%s Sex:%s\r\n", stude…...
Linux进阶系列(四)——awk、sed、端口管理、crontab
目录 1. 写在前面2. awk —— 强大的文本处理工具2.1 awk 概述2.2 awk 脚本结构2.3 awk 的内置变量2.4 awk 的高级用法2.5 awk实践 3. sed —— 流式文本编辑器3.1 sed 的基本语法3.2 sed 常用命令3.3 sed 的高级用法 4. Linux 端口管理4.1 端口的概念4.2 查看端口状态4.3 开放…...
利用Metasploit进行信息收集与扫描
Metasploit之信息收集和扫描 在本文中,我们将学习以下内容 使用Metasploit被动收集信息 使用Metasploit主动收集信息 使用Nmap进行端口扫描 使用db_nmap方式进行端口扫描 使用ARP进行主机发现 UDP服务探测 SMB扫描和枚举 SSH版本扫描 FTP扫描 SMTP枚举 …...
基于Pytorch框架的深度学习MODNet网络精细人像分割系统源码
第一步:准备数据 人像精细分割数据,可分割出头发丝,为PPM-100开源数据 第二步:搭建模型 MODNet网络结构如图所示,主要包含3个部分:semantic estimation(S分支)、detail prediction…...
Go语言中的并发编程
Go语言中的并发编程Go语言中的并发编程主要依赖于两个核心概念:goroutine 和 channel。1. Goroutinegoroutine 的特点结束 goroutine2. Channel创建 Channel发送和接收数据Channel 的类型使用 select 语句简单的多个 goroutine使用 WaitGroup 等待所有 goroutine 完…...
python学习笔记(3)——控制语句
控制语句 我们在前面学习的过程中,都是很短的示例代码,没有进行复杂的操作。现在,我们将开始学习流程控制语句。 前面学习的变量、数据类型(整数、浮点数、布尔)、序列(字符串、列表、元组、字 典、集合&am…...
关系数据库设计之Armstrong公理详解
~犬📰余~ “我欲贱而贵,愚而智,贫而富,可乎? 曰:其唯学乎” 一、Armstrong公理简介 Armstrong公理是一组在关系数据库理论中用于推导属性依赖的基本规则。这些公理是以著名计算机科学家威廉阿姆斯特朗&…...
【Geoserver使用】SRS处理选项
文章目录 前言一、Geoserver的三种SRS处理二、对Bounding Boxes计算的影响总结 前言 今天来看看Geoserver中发布图层时的坐标参考处理这一项。根据Geoserver官方文档,坐标参考系统 (CRS) 定义了地理参考空间数据与地球表面实际位置的关系。CRS 是更通用的模型&…...
python里面的单引号和双引号的区别
在Python中,单引号(‘’)和双引号(“”)在大多数情况下是等价的,没有本质区别。它们都用于创建字符串。以下是一些关键点: 功能相同: 两者都可以用来定义字符串,例如&…...
为什么不要在循环,条件或嵌套函数中调用hooks
为什么不要在循环,条件或嵌套函数中调用hooks 前言useState Hook 的工作原理具体实现1、初始化2、第一次渲染3、后续渲染4、事件处理简单代码实现 为什么顺序很重要Bad Component 第一次渲染Bad Component 第二次渲染 总结 前言 自从 React 推出 hooks 的 API 后&a…...
将成功请求的数据 放入apipost接口测试工具,发送给后端后,部分符号丢失
将成功请求的数据 放入apipost接口测试工具,发送给后端后,部分符号丢失 apipost、接口测试、符号、丢失、错乱、变成空格背景 做CA对接,保存CA系统的校验数据,需要模仿前端请求调起接口,以便测试功能完整性。 问题描…...
N诺计算机考研-错题
B A.LLC,逻辑链路控制子层。一个主机中可能有多个进程在运行,它们可能同时与其他的一些进程(在同一主机或多个主机中)进行通信。因此在一个主机的 LLC子层的一个服务访问点,以便向多个进程提供服务。B.MAC地址,称为物理地址、硬件地址,也称为局域网地址,用来定义网络设…...
vue3 数字滚动组件封装
相关参考文献 干货满满!如何优雅简洁地实现时钟翻牌器(支持JS/Vue/React) Vue3 插件方式 安装插件: countup.js 封装组件: components/count-up/index.js <template><div class="countup-wrap"><slot name="prefix"></slot&g…...
如何确保消息只被消费一次:Java实现详解
引言 在分布式系统中,消息传递是系统组件间通信的重要方式,而确保消息在传递过程中只被消费一次是一个关键问题。如果一个消息被多次消费,可能会导致业务逻辑重复执行,进而产生数据不一致、错误操作等问题。特别是在金融、电商等…...
Web3技术在元宇宙中的应用:从区块链到智能合约
随着元宇宙的兴起,Web3技术正逐渐成为其基础,推动着数字空间的重塑。元宇宙不仅是一个虚拟世界,它还代表着一个由去中心化技术驱动的新生态系统。在这个系统中,区块链和智能合约发挥着至关重要的作用,为用户提供安全、…...
关于QSizeGrip在ui界面存在布局的情况下的不显示问题
直接重写resizeEvent你会发现:grip并没有显示 void XXXXX::resizeEvent(QResizeEvent *event) {QWidget::resizeEvent(event);this->m_sizeGrip->move(this->width() - this->m_sizeGrip->width() - 3,this->height() - this->m_sizeGrip->…...
开始场景的制作+气泡特效的添加
3D场景或2D场景的切换 1.新建项目时选择3D项目或2D项目 2.如下图操作: 开始前的固有流程 按照如下步骤进行操作,于步骤3中更改Company Name等属性: 本案例分辨率可以如下设置,有能力者可根据需要自行调整: 场景制作…...
位运算--(二进制中1的个数)
位运算是计算机科学中一种高效的操作方式,常用于处理二进制数据。在Java中,位运算通常通过位移操作符和位与操作符实现。 当然位运算还有一些其他的奇淫巧计,今天介绍两个常用的位运算方法:返回整数x的二进制第k位的值和返回x的最…...
conda相比python好处
Conda 作为 Python 的环境和包管理工具,相比原生 Python 生态(如 pip 虚拟环境)有许多独特优势,尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处: 一、一站式环境管理:…...
【Python】 -- 趣味代码 - 小恐龙游戏
文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...
服务器硬防的应用场景都有哪些?
服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式,避免服务器受到各种恶意攻击和网络威胁,那么,服务器硬防通常都会应用在哪些场景当中呢? 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...
Golang dig框架与GraphQL的完美结合
将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用,可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器,能够帮助开发者更好地管理复杂的依赖关系,而 GraphQL 则是一种用于 API 的查询语言,能够提…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...
苍穹外卖--缓存菜品
1.问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大 2.实现思路 通过Redis来缓存菜品数据,减少数据库查询操作。 缓存逻辑分析: ①每个分类下的菜品保持一份缓存数据…...
微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
今日科技热点速览
🔥 今日科技热点速览 🎮 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售,主打更强图形性能与沉浸式体验,支持多模态交互,受到全球玩家热捧 。 🤖 人工智能持续突破 DeepSeek-R1&…...
