前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【STL】string的使用

【STL】string的使用

作者头像
诺诺的包包
发布2023-10-15 13:57:01
1500
发布2023-10-15 13:57:01
举报
文章被收录于专栏:个人笔记总结个人笔记总结

放在专栏【C++知识总结】,会持续更新,期待支持🌹

STL简介

STL的诞生

STL为英文Standard Template Library的缩写,译为标准模板库是C++标准库的重要组成部分

  • 长久以来,软件届一直希望建立一种可重复运用的东西。所谓的泛型思想以及面向对象最主要的目的就是为了复用性的提升
  • 复用性需要建立在某种标准之上,而数据结构与算法却迟迟未能有一套标准,导致大量程序员被迫从事大量重复的工作
  • STL的出现实现了在数据结构与算法之间建立一套标准,并降低其耦合度。

STL具有多个版本:HP版本、P.J.版本、RW以及SGI版本,由于SGI版本在命名风格以及编程风格方面可读性更佳,所以后续一些学习都将参考该版本。

STL的组成部分

STL由六大组件构成,分别为:容器、算法、迭代器、仿函数、配接器、空间配置器。彼此之间可以组合套用。

接下来,我们学习的就是关于string相关使用。

string类

string简介

在C语言中,有整形、字符类型、浮点型等,但是=并没有字符串类型。而对于字符串,C语言中通常都是使用字符指针或字符数组来存储。但是字符指针指向的是字符常量,不可被修改,而字符数组又面临数组越界等情况。并且也不符合面向对象的思想。C++针对于此,string诞生。

如果我们仔细观察的话,就会发现,实际上string类是basic_string模板类使用char来实例化出来的一个类。对于basic_string模板类,这里不做过多介绍,有兴趣的可以搜寻相关文档。

 string的使用

(不要忘记包含头文件<string>

构造函数

string的构造函数有很多接口,没必要各个都掌握,否则学习起来会比较繁琐。(这也是C++被吐槽的原因之一,这么多接口,咋可能都记住)。我们只需要掌握其中一些即可,其余的可以翻阅文档。如下:

string提供的构造函数

功能说明

string()

构造空的string类对象,即空字符串

string(const char* str)

用C语言格式的字符串,来构造一个string类对象

string(size_t n,char c)

用n个字符,来构造一个string类对象

string(const string& str)

拷贝构造

具体使用如下

代码语言:javascript
复制
#include<iostream>
#include<string>//头文件
//using namespace std;//也可以直接展开命名空间,后面就不需要声明string属于std这个域了

int main()
{
	std::string s1;               //空字符串构造
	std::string s2("hello world");//用C格式字符串构造
	std::string s3(10, 'a');      //10个a字符,来构造一个string对象

	std::string s4(s3);           //拷贝构造

	std::cout << "s1: " << s1 << std::endl;
	std::cout << "s2: " << s2 << std::endl;
	std::cout << "s3: " << s3 << std::endl;
	std::cout << "s4: " << s4 << std::endl;

	return 0;
}

运行结果如下:

 容量相关

string提供的有关容量的函数

功能说明

size

返回有效字符串的长度

length

与size相同,推荐使用size

capacity

空间总大小

empty

字符串是否为空,空返回true,非空返回false

clear

清空有效字符

reserve

进行扩容,改变的是总空间的大小,直接影响capacity

resize

对有效字符进行扩容,改变的是size的大小,可能会间接影响capacity

size与capacity

我们来看,size为有效字符的长度大小,capacity为总空间大小。两者并不相同。如下图:

clear与empty就不用多说了,这里需要注意的是,clear只是将有效字符的第一个字符设置成\0,不会改变总空间的大小,但是会影响size,因为size计算的是有效字符串的长度,遇到\0结束。

 reserve与resize

两者都是扩容,不同的是,reserve是改变总空间的大小,而resize改变的是有效字符的大小。如下:

reserve进行空间扩容时,采用异地扩容 。

可能有人也想我这么皮,用reserve缩容,看看会发生啥,如下:reserve不会进行缩容。 

resize

 resize则是将有效字符扩容成n个,如果不指定c,则默认用0来填充剩余空间,如果扩容后的有效字符超过了总空间的大小,则总空间会自动扩容。

VS与Linux扩容策略对比

如果细心的话,就会发现,我们在VS中进行扩容时,一开始时数据存放在15大小的_buf数组中,后来在扩容超过该数组空间时,才又开辟额外空间。

我们可以写如下代码来进行验证一下:

代码语言:javascript
复制
int main()
{
	string s;
	size_t sz = s.capacity();
	cout << "making s grow:\n";
	for (int i = 0; i < 100; ++i)
	{
		s.push_back('c');
		if (sz != s.capacity())
		{
			sz = s.capacity();
			cout << "capacity changed: " << sz << '\n';
		}
	}
}

VS扩容策略

  1. 数据size小于15时,存放在buf数组中。
  2. 当数据大于15时,先2倍扩容,后续再进行扩容时以1.5倍进行扩容

Linux扩容策略

  1. 不存在buf数组,默认空间大小为0
  2. 从1开始,严格遵守2倍扩容

实际上VS的策略更加合理一些,因为实际中string的大小不会太大,给个15大小的数组,一般是够用的。而Linux频繁的扩容,会导致内存碎片问题,因此VS策略会更加合理一些。另外,我们在使用string时,如果能提前计算出所需要的空间,直接reserve提前扩容,会提高一定的运行效率。

string类对象的访问以及遍历

1.[]下标访问

由于string对[]进行重载,所以支持[]进行访问,就像数组下标一样。使用也很简单,如下:

 2.迭代器

什么是迭代器?

首先我们来谈一谈迭代器是个啥东西,迭代器既然可以作为六大组件之一,那么一定有它的过人之处。我们先来看一下迭代器在书中《STL源码剖析》的定义:

 简单来说,就是用一个对象,来模拟一个指针的行为,从而实现对各个容器中成员的访问

在string,或者vector(实际上就是一个数组)中,我们可以将它简单的想象成一个指针,指针++,表示访问下一个元素的地址,*解引用,表示访问该位置的成员。(当然,它并不是一个指针,因为像链表、树形结构中,各个元素之间的地址并不是紧密相连,后续遇到再仔细讲解,这里就简单理解为指针)。

回归正题,这里用迭代器进行成员访问,如下:

 reverse_iterator为反向迭代器,顾名思义可以倒着访问,使用如下:

当然,string还存在const_iterator与const_reverse_iterator。看名字也可以看出来,不可对成员进行修改。用法与上面一样。具体迭代器的细节将放在后面list章节再继续探讨。

3、范围for

范围for的底层实际上是迭代器,用法也很简单,在前面章节已经讲解过。

string类对象的修改操作

插入与删除

我们查阅官方文档,发现有大量的接口供我们使用,这里我就只讲某个函数的其中一个来讲解(实在是太多,不得不吐槽)

string支持的插入函数

功能含义

push_back

尾插字符

append

追加字符串

+=

追加字符串

insert

插入操作

这些接口都很简单,用法如下:

 删除erase,用法也很简单,这里就不做过多演示。大家可以自己去试着玩一玩。

查找find

 find默认从0下标开始,查找一个字符或者字符串,找到后返回该字符所在的下标。找不到返回npos。npos实际上就是-1,而size_t是无符号整形,所以这里的npos代表整形最大值。用法如下:

 string提供的接口实在是太多了,大家自行可以去网站https://cplusplus.com/reference/查找。这里我就不做相关演示了。

string的模拟实现

为了能更好的理解底层的实现,我们可以参照STL中string的源码实现,来自己实现一个简单的string,从而加深对string的理解。源码可能会复杂难懂,推荐书籍《STL源码剖析》,里面会有一些关键标注。当然我们在实现时没必要面面俱到,实现大体即可。

以下链接内为我个人实现的一个简单string,内部含有个人注释,如有不解,可联系交谈:

string类的模拟实现练习 · ede63fa · 齐敦炎/C++学习所用仓库 - Gitee.com

end.

生活原本沉闷,但跑起来就会有风!🌹

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-10-11,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • STL简介
    • STL的诞生
      • STL的组成部分
      • string类
        • string简介
          •  string的使用
            • 构造函数
            •  容量相关
          • VS与Linux扩容策略对比
            • string类对象的访问以及遍历
          • string类对象的修改操作
            • 插入与删除
            • 查找find
        • string的模拟实现
        相关产品与服务
        容器服务
        腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档