【C++修行之道】string类的使用

目录

一.C语言中的字符串

二、标准库中的string类 (了解)

2.1 string类(了解)

2.2 帮助文档阅读

三、 string类的常用接口说明

3.1 string类对象的常见构造 

3.2 string类对象的容量操作

3.3 string类对象的访问及遍历操作

字符串类的简单实现

3.4 string类对象的修改操作

3.5 string类非成员函数

3.6 vs和g++下string结构的说明

四、string操作的代码示例


一.C语言中的字符串

C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数, 但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可 能还会越界访问。

二、标准库中的string(了解)

2.1 string(了解)

1. 字符串是表示字符序列的类

2. 标准的字符串类提供了对此类对象的支持,其接口类似于标准字符容器的接口,但添加了专门用于操作单字节字符字符串的设计特性。

3. string类是使用char(即作为它的字符类型,使用它的默认char_traits和分配器类型(关于模板的更多信息,请参阅basic_string)。

4. string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,用char_traits 和allocator作为basic_string的默认参数(根于更多的模板信息请参考basic_string)。

5. 注意,这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。

C++中对于string的定义是:typedef basic_string string;

  • 使用 typedef 关键字将 basic_string 类定义为一个新的类型 string
  • 这意味着,以后在代码中提到 string 时,其实是在引用 basic_string 类。

也就是说C++中的string类是一个泛型类,由模板而实例化的一个标准类,本质上不是一个标准数据类型。

至于为什么不直接用String标准数据类型而用类是因为编码

每个国家的语言不同 比如说英语使用26个英文字母基本就能表述所有的单词 但是对于中文的字符呢?就要用其他编码方式啊(比如说utf-8)

总结:

  1. string是表示字符串的字符串类
  2. 该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。
  3. string在底层实际是:basic_string模板类的别名,typedef basic_string<char, char_traits, allocator> string;
  4. 不能操作多字节或者变长字符的序列。
  5. 在使用string类时,必须包含#include头文件以及using namespace std

2.2 帮助文档阅读

(1) empty string constructor (default constructor)

        Constructs an empty string, with a length of zero characters.

(2) copy constructor

        Constructs a copy of str.

(3) substring constructor

        Copies the portion of str that begins at the character position pos and spans len characters         (or until the end of str, if either str is too short or if len is string::npos).

(4) from c-string

        Copies the null-terminated character sequence (C-string) pointed by s.

(5) from buffer

        Copies the first n characters from the array of characters pointed by s.

(6) fill constructor

        Fills the string with n consecutive copies of character c.

(7) range constructor

        Copies the sequence of characters in the range [first,last), in the same order.

(1) 空字符串构造函数(默认构造函数)       

        构造一个空字符串,长度为零个字符。

(2) 拷贝构造函数         

        构造一个 str 的副本。

(3) 子字符串构造函数         

        复制 str 从字符位置 pos 开始并跨越 len 个字符的部分

        (如果 str 太短或 len 是 string::npos,则复制到 str 的末尾)。

(4) 从 C 风格字符串构造         

        复制由 s 指向的以 null 结尾的字符序列(C 字符串)。

(5) 从字符串序列构造         

        复制由 s 指向的字符数组中的前 n 个字符。 

(6) 填充构造函数        

        用 n 个字符 c 的连续副本填充字符串。

(7) 范围构造函数         

        复制范围 [first,last) 中的字符序列,顺序保持不变。

三、 string类的常用接口说明

3.1 string类对象的常见构造 

构造函数函数名称功能说明
string()空字符串构造函数(重点)构造空的 string 类对象,即空字符串
string(const char* s)从 C-string 构造(重点)用 C-string 来构造 string 类对象
string(size_t n, char c)填充构造函数string 类对象中包含 n 个字符 c
string(const string& s)拷贝构造函数(重点)用一个已有的 string 对象 s 构造新的 string 对象
void test01()
{// 常用string s1;string s2("hello world");string s3(s2);// 不常用, 了解string s4(s2, 3, 5);string s5(s2, 3);string s6(s2, 3, 30);string s7("hello world", 5);string s8(10, 'x');cout << s1 << endl;cout << s2 << endl;cout << s3 << endl;cout << s4 << endl;cout << s5 << endl;cout << s6 << endl;cout << s7 << endl;cout << s8 << endl;cin >> s1;cout << s1 << endl;
}

3.2 string类对象的容量操作

函数名称功能说明
size(重点)返回字符串有效字符长度
length返回字符串有效字符长度
capacity返回空间总大小
empty(重点)检测字符串是否为空串,若为空返回 true,否则返回 false
clear(重点)清空有效字符
reserve(重点)为字符串预留空间
resize(重点)将有效字符的个数改为 n 个,多出的空间用指定字符填充

注意:

1. size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一

致,一般情况下基本都是用size()。

2. clear()只是将string中有效字符清空,不改变底层空间大小。

3. resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字 符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的 元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。

4. reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于 string的底层空间总大小时,reserver不会改变容量大小。

3.3 string类对象的访问及遍历操作

函数名称功能说明
operator[](重点)返回 pos 位置的字符,用于 const string 类对象的调用
beginendbegin 获取一个字符的迭代器,end 获取最后一个字符下一个位置的迭代器
rbeginrendrbegin 获取一个反向迭代器指向最后一个字符,rend 获取一个反向迭代器指向第一个字符前一个位置
范围 forC++11 支持更简洁的范围 for 新遍历方式

字符串类的简单实现

class String
{
public:// 引用返回// 1.减少拷贝// 2.修改返回对象char& operator[](size_t i){assert(i < _size)return _str[i];}
private:char* _str;size_t _size;size_t _capacity;
};

其中operator[]操作符重载允许通过下标访问和修改字符串中的字符。通过返回字符的引用,可以避免不必要的拷贝并直接修改字符串内容,同时通过assert保证了安全性。

void test02()
{string s1("hello world");cout << s1.size() << endl;// 不计算'\0'// cout << s1.length() << endl;// 访问每个字符并令其++for (size_t i = 0; i < s1.size(); i++){s1[i]++;}cout << endl;s1[0] = 'x';// 越界会被检查// s1[20];// 访问每个字符for (size_t i = 0; i < s1.size(); i++){cout << s1.operator[](i) << " ";cout << s1[i] << " ";}cout << endl;const string s2("hello world");// const 不能修改// s2[0] = 'x';
}

通过重载的 operator[] 访问和修改字符串中的字符,同时也演示了对常量字符串对象进行修改时的限制。代码还包含了越界访问检查,通过 assert 保证安全性。

3.4 string类对象的修改操作

函数名称功能说明
push_back在字符串末尾插入字符 c
append在字符串末尾追加一个字符串
operator+=在字符串末尾追加字符串 str(重点)
c_str返回 C 格式字符串(重点),用于与其他 C/C++ 函数交互,返回的字符串是临时的,不应被修改
find + npos从字符串 pos 位置开始往后找字符 c,返回该字符在字符串中的位置,找不到时返回 string::npos(重点)
rfind从字符串 pos 位置开始往前找字符 c,返回该字符在字符串中的位置(注意:rfind 通常用于查找子字符串,而非单个字符 c 的向前查找,对于单个字符,直接使用 find 从末尾开始查找即可)
substr在字符串中从 pos 位置开始,截取 n 个字符,然后将其返回

注意:

1. 在string尾部追加字符时,s.push_back(c) / s.append(1, c) / s += 'c'三种的实现方式差不多,一般情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。

2. 对string操作时,如果能够大概预估到放多少字符,可以先通过reserve把空间预留好。

void test03()
{// 构造string s1("hello world");// 隐式类型转换string s2 = "hello world";const string& s3 = "hello world";// 引用的临时对象, 临时对象具有常性string s4("hello world");push_back(s4);push_back("hello world");
}

3.5 string类非成员函数

函数/运算符功能说明
operator+字符串连接运算符,但尽量少用,因为传值返回导致深拷贝,效率低
operator>> (重点)输入运算符重载,用于从输入流中提取字符串到字符串对象中
operator<< (重点)输出运算符重载,用于将字符串对象的内容发送到输出流中
getline (重点)从输入流中获取一行字符串,直到遇到换行符('\n'),不包括换行符
relational operators (重点)大小比较运算符重载,包括 <<=>>===!=,用于比较两个字符串对象的大小

上面的几个接口大家了解一下,下面的OJ题目中会有一些体现他们的使用。string类中还有一些其他的 操作,这里不一一列举,大家在需要用到时不明白了查文档即可。

3.6 vs和g++下string结构的说明

注意:下述结构是在32位平台下进行验证,32位平台下指针占4个字节。

vsstring的结构

string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,联合体用来定义string中字

符串的存储空间

  • 当字符串长度小于16时,使用内部固定的字符数组来存放
  • 当字符串长度大于等于16时,从堆上开辟空间
union _Bxty
{ // storage for small buffer or pointer to larger onevalue_type _Buf[_BUF_SIZE];pointer _Ptr;char _Alias[_BUF_SIZE]; // to permit aliasing
} _Bx;

这种设计也是有一定道理的,大多数情况下字符串的长度都小于16,那string对象创建好之后,内

部已经有了16个字符数组的固定空间,不需要通过堆创建,效率高。

其次:还有一个size_t字段保存字符串长度,一个size_t字段保存从堆上开辟空间总的容量

最后:还有一个指针做一些其他事情。

故总共占16+4+4+4=28个字节。

g++string的结构

G++下,string是通过写时拷贝实现的,string对象总共占4个字节,内部只包含了一个指针,该指

针将来指向一块堆空间,内部包含了如下字段:

  • 空间总大小
  • 字符串有效长度
  • 引用计数
struct _Rep_base
{size_type _M_length;size_type _M_capacity;_Atomic_word _M_refcount;
};
  • 指向堆空间的指针,用来存储字符串。

四、string操作的代码示例

void test04()
{string s1("hello world");// 遍历方式1: 下标 + []for (size_t i = 0; i < s1.size(); i++){cout << s1[i] << " ";}cout << endl;// 遍历方式2: 迭代器string::iterator it1 = s1.begin();while (it1 != s1.end()){*it1 += 3;cout << *it1 << " ";++it1;}cout << endl;// 遍历方式3: 范围for// 底层实现方式都是迭代器实现for (auto e : s1){e++;cout << e << " ";}cout << endl;//cout << typeid(it1).name() << endl;/*list<int> lt1;lt1.push_back(1);lt1.push_back(2);lt1.push_back(3);list<int>::iterator it = lt1.begin();while (it != lt1.end()){cout << *it << " ";++it;}cout << endl;for (auto e : lt1){cout << e << " ";}cout << endl;*/
}

//template<class T>
//struct ListNode
//{
//	ListNode <T>* _next;
//	ListNode <T>* _prev;
//	T _data;
//};
//
//template<class T>
//class list
//{
//private:
//	ListNode<T>* _head;
//};void test05()
{const string s1("hello world");//string::const_iterator it1 = s1.begin();auto it1 = s1.begin();while (it1 != s1.end()){// 指向的内容不能修改// *it1 += 3;cout << *it1 << " ";++it1;// const_iterator 本身可以++}cout << endl;/*string::iterator it1 = s1.begin();while (it1 != s1.end()){*it1 += 3;cout << *it1 << " ";++it1;}cout << endl;*/
}

void test06()
{const string s1("hello world");//string::const_iterator it1 = s1.begin();auto it1 = s1.begin();while (it1 != s1.end()){// 不能修改//*it1 += 3;cout << *it1 << " ";++it1;}cout << endl;//string::const_reverse_iterator cit1 = s1.rbegin();auto cit1 = s1.rbegin();while (cit1 != s1.rend()){// 不能修改//*cit1 += 3;cout << *cit1 << " ";++cit1;}cout << endl;string s2("hello world");string::reverse_iterator it2 = s2.rbegin();//auto it2 = s2.rbegin();while (it2 != s2.rend()){//*it2 += 3;cout << *it2 << " ";++it2;}cout << endl;
}

void test07()
{string s1("hello world");cout << s1 << endl;// s1按字典序排序// 排序一部分sort(s1.begin(), s1.end());// 第一个和最后一个参与排序sort(++s1.begin(), --s1.end());// 前五个排序  [0, 5)// sort(s1.begin(), s1.begin() + 5);cout << s1 << endl;
}

void test08()
{string s1("hello world");cout << s1 << endl;s1.push_back('x');cout << s1 << endl;s1.append(" yyyy!!");cout << s1 << endl;s1 += 'y';s1 += "zzzzz";cout << s1 << endl;
}

void test09()
{string s1("hello world");cout << s1 << endl;s1.assign("111111");cout << s1 << endl;// 慎用, 因为效率不高 -> O(N)// 实践中需求也不高string s2("hello world");s2.insert(0, "xxxx");cout << s2 << endl;char ch = 'z';s2.insert(0, 1, ch);s2.insert(0, 1, 'y');cout << s2 << endl;// 没有这个版本的头插// s2.insert(0, 'y');// cout << s2 << endl;s2.insert(s2.begin(), 'y');cout << s2 << endl;s2.insert(s2.begin(), s1.begin(), s1.end());cout << s2 << endl;
}

void test10()
{string s1("hello world");cout << s1 << endl;// 效率不高, 慎用, 和inset类似s1.erase(0,1);// 头删cout << s1 << endl;// s1.erase(5);s1.erase(5,100);cout << s1 << endl;string s2("hello world");s2.replace(5, 1, "%20");// 字符串的替换cout << s2 << endl;// 不建议这样使用,因为要挪动数据string s3("hello world hello char");for (size_t i = 0; i < s3.size();){if (s3[i] == ' '){s3.replace(i, 1, "%20");i += 3;}else {i++;}}cout << s3 << endl;string s4("hello world hello double");string s5;for (auto ch : s4){if (ch != ' '){s5 += ch;}else {s5 += "%20";}}cout << s5 << endl;
}

今天就先到这了!!!、

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

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

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

相关文章

【论文阅读笔记】ASPS: Augmented Segment Anything Model for Polyp Segmentation

1.论文介绍 ASPS: Augmented Segment Anything Model for Polyp Segmentation ASPS&#xff1a;用于息肉分割的扩展SAM模型 2024年 arxiv Paper Code 2.摘要 息肉分割在结直肠癌诊断中起着至关重要的作用。最近&#xff0c;Segment Anything Model(SAM)的出现利用其在大规模…

视频压缩软件哪个压缩最小,视频用什么软件压缩最小

在数字媒体时代&#xff0c;视频内容的生产与分享已成为生活常态。但随之而来的问题就是&#xff0c;大视频文件占用过多存储空间&#xff0c;上传和分享也变得不便。本文将为你揭示如何将视频压缩到最小&#xff0c;同时保持画质清晰。让我们一起探索吧&#xff01; 下载并文件…

剪辑抽帧技巧有哪些 剪辑抽帧怎么做视频 剪辑抽帧补帧怎么操作 剪辑抽帧有什么用 视频剪辑哪个软件好用在哪里学

打破视频节奏&#xff0c;让作品告别平庸。抽帧剪辑可以改变视频叙事节奏&#xff0c;人为制造冲突、转折、卡顿的效果。这种剪辑方式&#xff0c;不仅可以推进剧情发展&#xff0c;还能吸引观众的注意力&#xff0c;有效防止观影疲劳。有关剪辑抽帧技巧有哪些&#xff0c;剪辑…

Typora篇-忍痛开启

语雀专业会员即将到期, 我看着99元的学费款, 我决定重新用回Typora。 虽然里面有一些文件但是我还是舍不得ಥ_ಥ 99元巨款。 下面开启我的Typora整活历程&#xff0c; 大家有什么好用的插件快捷方式一起来分享啊。

MyBatis框架学习笔记(二):原生API 的调用 和 注解的使用

1 MyBatis原生API 1.1 原生API 快速入门需求 在笔记一案例的基础上将增删改查&#xff0c;使用 MyBatis 原生的 API 完成&#xff0c;就是直接通过SqlSession 接口的方法来完成 1.2 原生API 快速入门-代码实现 创建 src\test\java\com\hspedu\mapper\MyBatisNativeTest.jav…

uni-app 封装http请求

1.引言 前面一篇文章写了使用Pinia进行全局状态管理。 这篇文章主要介绍一下封装http请求&#xff0c;发送数据请求到服务端进行数据的获取。 感谢&#xff1a; 1.yudao-mall-uniapp: 芋道商城&#xff0c;基于 Vue Uniapp 实现&#xff0c;支持分销、拼团、砍价、秒杀、优…

实用性提升百分之一百!!!【ONLYOFFICE 8.1版本】全方位深度性能测评

目录 【ONLYOFFICE 8.1 版本】全方位深度性能测评 一、界面与用户体验 二、文字处理功能 表格处理功能 演示文稿功能 协作与共享功能 性能与稳定性 总结 【ONLYOFFICE 8.1 版本】全方位深度性能测评 在当今数字化办公的时代&#xff0c;办公软件的选择对于提高工作效率和…

selenium处理cookie问题实战

1. cookie获取不完整 需要进入的资损平台(web)首页&#xff0c;才会出现有效的ctoken等信息 1.1. 原因说明 未进入指定页面而获取的 cookie 与进入页面后获取的 cookie 可能会有一些差异&#xff0c;这取决于网站的具体实现和 cookie 的设置方式。 通常情况下&#xff0c;一些…

【解决Windows11系统Windows Hello不能使用的问题】

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、Windows Hello是什么&#xff1f;二、使用步骤1.购买一个摄像头2.开始配置 三、异常解决1.内置管理员不能使用2.没找到合适的摄像头3.摄像头需要专用驱动4.…

原创作品——教育课程界面设计

教育行业UI界面设计需直观易懂&#xff0c;确保学习者能迅速上手&#xff0c;减少认知负担。布局清晰&#xff0c;导航便捷&#xff0c;功能按钮和图标设计应符合教育场景&#xff0c;直接支持学习目标的达成&#xff0c;促进高效学习体验。 通过丰富的互动元素&#xff08;如拖…

博途通讯笔记1:1200与1200之间S7通讯

目录 一、添加子网连接二、创建PUT GET三、各个参数的意义 一、添加子网连接 二、创建PUT GET 三、各个参数的意义

代码随想录(day1)二分法

if语句的基本语法 if 要判断的条件: 条件成立的时候&#xff0c;要做的事举例&#xff1a; if nums[middle]<target:leftmiddle1 while语句的基本语法&#xff1a; while 判断条件(condition)&#xff1a;执行语句(statements)举例&#xff1a; while left<right:midd…

2通道音频ADC解码芯片ES7243L、ES7243E、ES7243,用于低成本实现模拟麦克风转换为IIS数字话筒

前言&#xff1a; 音频解码芯片某创参考价格&#xff1a; ES7243L 500&#xff1a;&#xffe5;1.36 / 个 ES7243E 500&#xff1a;&#xffe5;1.66 / 个 ES7243 500&#xff1a; &#xffe5;1.91 / 个 其中ES7243L工作电压为1.8V&#xff0c;与其他两款的3.3V工作电压不同&…

日本IT-SIER/SES的区别详情、契约形态等

一、SLER 主要的服务内容就是“帮客人开发系统或是各种APP&#xff0c;并在指定期间内交货&#xff0c;交货后也会持续进行运维等售后服务”。 客人很广泛&#xff0c;小到普通的服务业商家&#xff08;餐饮店/服饰店/美容业/电商&#xff09;大到各种公共/政府机关&#xff…

【面试题】串联探针和旁挂探针有什么区别?

在网络安全领域中&#xff0c;串联探针和旁挂探针&#xff08;通常也被称为旁路探针&#xff09;是两种不同部署方式的监控设备&#xff0c;它们各自具有独特的特性和应用场景。以下是它们之间的主要区别&#xff1a; 部署方式 串联探针&#xff1a;串联探针一般通过网关或者…

第一百四十九节 Java数据类型教程 - Java子字符串、字符串转换

Java数据类型教程 - Java子字符串 获取子字符串 我们可以使用substring()方法来获取字符串的子部分。 我们可以将开始索引作为参数&#xff0c;并返回一个从开始索引开始到字符串结尾的子串。 我们还可以将开始索引和结束索引作为参数。 它返回从开始索引开始的子字符串和小…

Windows右键没有新建Word、PPT与Excel的解决方法

本文介绍在Windows电脑中&#xff0c;右键与资源管理器的“新建”选项中&#xff0c;都没有新建Word、PPT或Excel文件的解决方法。 最近&#xff0c;发现一台重装了系统与Office的电脑中&#xff0c;无论是桌面上与资源管理器中的右键&#xff0c;还是资源管理器左侧顶部的“新…

手写简单模拟mvc

目录结构&#xff1a; 两个注解类&#xff1a; Controller&#xff1a; package com.heaboy.annotation;import java.lang.annotation.*;/*** 注解没有功能只是简单标记* .RUNTIME 运行时还能看到* .CLASS 类里面还有&#xff0c;构建对象久没来了&#xff0c;这个说明…

C++编译链接原理

从底层剖析程序从编译到运行的整个过程 三个阶段 一、编译阶段二、链接阶段三、运行阶段 为了方便解释&#xff0c;给出两端示例代码&#xff0c;下面围绕代码进行实验&#xff1a; //sum.cpp int gdata 10; int sum(int a,int b) {return ab; }//main.cpp extern int gdata…

初始redis:在Ubuntu上安装redis

1.先切换到root用户 使用su命令切换到root 2.使用apt命令来搜索redis相关的软件包 命令&#xff1a;apt search redis 3.下载redis 命令&#xff1a; apt install redis 在Ubuntu 20.04中 &#xff0c;下载的redis版本是redis5 4.查看redis状态 命令&#xff1a; netst…