【C++深入浅出】STL之string用法详解

news/2024/6/3 18:29:46 标签: c++, 开发语言, stl, 容器, 数据结构, string


目录

一. 前言

二. STL概要

2.1 什么是STL

2.2 STL的六大组件

2.3 STL的缺陷

string%E7%B1%BB%E6%A6%82%E8%BF%B0-toc" style="margin-left:40px;">三. string类概述

string%E7%B1%BB-toc" style="margin-left:80px;">3.1 什么是string

string%E7%B1%BB-toc" style="margin-left:80px;">3.2 为什么要使用string

string%E7%B1%BB%E7%9A%84%E4%BD%BF%E7%94%A8-toc" style="margin-left:40px;">四. string类的使用

4.1 包含头文件

4.2 构造函数

4.3 赋值运算符重载

4.4 容量操作

4.5 访问/遍历操作

4.6 查找修改操作

4.7 子串操作

​4.8 非成员函数


一. 前言

        经历了前面漫长且痛苦的学习,相比各位已经体会到了C++的魅力了叭 不要怕,学习完了模板之后,下面我们将进入STL的学习。相信你学完了STL之后,就会感受到使用C++是多么的顺畅,你甚至会不想回到使用C语言的时期,不信?就让我们拭目以待叭

二. STL概要

2.1 什么是STL

        把STL说得那么神,那STL究竟是什么呢?

        STL(standard template libaray - 标准模板库)C++标准库的重要组成部分。其不仅是一个可复用的组件库,而且是一个包罗数据结构与算法的软件框架。有了STL后,像先前我们写的链表、顺序表、排序算法等等常见的底层数据结构和算法我们就可以不用自己造轮子了,我们可以站在前人的肩膀上,快速的进行开发。

2.2 STL的六大组件

        STL主要由以下六大组件构成:

        在后续的学习中,我们会逐个接触到,到时我们在进行详细讲解。

2.3 STL的缺陷

        尽管STL功能十分强大,但也存在着一些缺陷,下面我们列出了几点:

1. STL库的更新太慢了。上一版靠谱是C++98,中间的C++03基本没有修订,到C++11出来已经相隔了13年,STL才进一步更新。
2. STL目前没有支持线程安全。并发环境下需要我们自己加锁。且锁的粒度是比较大的。
3. STL极度的追求效率,导致内部实现比较复杂。比如类型萃取,迭代器萃取。
4. STL的使用会有代码膨胀的问题,比如同时使用vector<int>,vector<double>,vector<string>时会生成多份代码,当然这是由于模板语法本身导致的。

string%E7%B1%BB%E6%A6%82%E8%BF%B0">三. string类概述

string%E7%B1%BB">3.1 什么是string

        string是C++标准库中用来表示字符序列的类,其中包含了许多关于操作单字节字符串的接口。string在底层实际上是basic_string模板类的别名,如下所示:

typedef basic_string<char, char_traits, allocator> string;

        可以看到,string是basic_string类模板的一个实例,其用char来实例化basic_string类模板,即string中存储的数据类型是char类型。故string只能用于处理单字节的序列,不能操作多字节或者变长字符的序列。

        string类的接口与常规容器的接口基本相同,只是添加了一些专门用来操作字符串的常规操作。

string%E7%B1%BB">3.2 为什么要使用string

        在C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太符合OOP(面向对象编程)的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。而C++标准库的string就不存在这些问题,极大程度上方便了用户的使用。

string%E7%B1%BB%E7%9A%84%E4%BD%BF%E7%94%A8">四. string类的使用

4.1 包含头文件

        首先,要使用string类,我们需要包含<string>头文件,并且由于string定义在命名空间std中,我们还需使用命名空间std,如下所示:

#include<string>
using namespace std;

接下来,我们将逐一介绍string的一些常用接口,由于string的接口以及重载版本非常多,详情读者可以参考以下链接:string - C++ Reference (cplusplus.com)icon-default.png?t=N7T8https://legacy.cplusplus.com/reference/string/string/?kw=string

4.2 构造函数

        string也是个类,使用类时我们首先关注的就是它的构造函数,string常见的构造函数如下所示:

        使用方式如下所示: 

void test_constructor()
{
	string str1; //无参构造
	string str2("abcd"); //C字符串构造
	string str3(str2); //拷贝构造
	string str4(str3, 1, 2); //子串构造
	string str5(str2.begin(), str2.begin()+3); //迭代器构造
}

这里有个小问题:在子串构造中,npos的含义又是什么呢?它的定义如下:

static const size_t npos = -1;

由于size_t是无符号整形,而-1在内存中二进制表示为全1,故将-1赋值给npos后npos实际上是2^32-1,是一个非常大的数。

string规定如果指定的长度len超过字符串的长度则只截取到字符串末尾,故将len的缺省值设置为npos的作用是当用户没有指定子串长度时默认截取到字符串末尾。

4.3 赋值运算符重载

        string重载了赋值运算符以便我们进行两个string对象之间的赋值,函数原型及说明如下所示:

         使用方式如下所示:

void test_assign()
{
	string str1("abcd");
	string str2 = "ab"; //注意,这里不是赋值重载,这里是定义并初始化str2,故调用的是构造函数
	str2 = str1; //这里才是赋值重载,用string对象赋值
	str2 = "aaaa"; //用一个C字符串赋值
	str2 = 'c'; //用一个字符赋值,此时str2的长度变为1
}

4.4 容量操作

        string类提供了许多与字符串容量相关的接口,如下表所示:

下面有几点注意事项

  1. size()与length()方法底层实现原理完全相同string出现得比STL早,那时string只有length方法,后面引入size()的原因是为了与其他STL容器的接口保持一致,一般情况下基本都是用size()
  2. clear只是把有效字符的个数清空,不会改变底层空间的大小。
  3. reverse()的作用是为string保留空间,其不改变有效元素个数,当reserve的参数n小于
    string的底层总容量大小时,reserver不会改变容量大小。
  4. 使用reverse()时,有时编译器实际分配的大小会比我们指定的空间,这取决于编译器,不过并不会造成什么影响,我们无需过多关心。

         使用方式如下所示: 

void test_capacity()
{
	string str1 = "hello world";
	cout << "size : " << str1.size() << endl; //输出有效字符长度
	cout << "lenfth : " << str1.size() << endl;
	cout << "capacity : " << str1.capacity() << endl; //输出当前的总容量
	str1.resize(5); //重新指定有效字符的长度
	cout << "size : " << str1.size() << endl;
	str1.reserve(20); //为字符串保留长度为20的空间
	cout << "capacity : " << str1.capacity() << endl;
	cout << "size : " << str1.size() << endl;
	str1.clear(); //清空有效字符
	cout << "empty : " << str1.empty() << endl; //判断字符串是否为空
	cout << "size : " << str1.size() << endl;
}

4.5 访问/遍历操作

        访问操作 

        遍历操作

        通过上面的操作,我们就可以轻松地对一个string对象进行遍历访问了,至于迭代器是什么,目前我们可以把它当做一个原生指针来使用,后面我们再进行详细介绍。元素访问和遍历的演示如下所示:

void  test_access()
{
	string str1("hello world");
	cout << "front : " << str1.front() << endl;
	cout << "back : " << str1.back() << endl;

	cout << "下标遍历:" << endl;
	for (int i = 0; i < str1.size(); i++)
	{
		cout << str1[i];
	}
	cout << endl;

	cout << "正向迭代器遍历:" << endl;
	string::iterator it = str1.begin();      //iterator类型在string的类域中,要加作用域限定符
	//auto it = str1.begin();  //也可以使用auto自动推导类型
	while (it != str1.end())
	{
		cout << *it;  //可以看做原生指针使用
		it++;
	}
	cout << endl;

	cout << "反向迭代器遍历:" << endl;
	string::reverse_iterator rit = str1.rbegin();      //反向迭代器的类型为reverse_iterator
	while (rit != str1.rend())
	{
		cout << *rit;  //可以看做原生指针使用
		rit++;
	}

	cout << "范围for遍历:" << endl; 
	for (auto e : str1)
	{
		cout << e;
	}
}

注意事项:

  1. 由于迭代器类型是在类域内进行重定义的,故我们要使用作用域限定符::指定其类域。
  2. 对于[ ]运算符和at方法,一般更喜欢使用[ ]进行访问,原因方括号更简便且顺手。
  3. 范围for遍历的实现原理实际上就是替换编译器会自动将范围for的代码替换为迭代器进行遍历。我们可以调试并对比二者的汇编验证:

4.6 查找修改操作

         和C语言的常量字符串相比,C++的string对象是可修改的,string类中的字符串是保存在堆空间上的。下面是string类中一些常见的查找和修改接口:

        查找操作

        使用方式如下所示: 

void test_find()
{
	string str1("hello world hello");
	string str2("hello");
	char ch = 'o';
	cout << "从前往后str2第一次出现的下标为:" << str1.find(str2) << endl;
	cout << "从前往后ch第一次出现的下标为:" << str1.find(ch) << endl;
	cout << "从后往前str2第一次出现的下标为:" << str1.rfind(str2) << endl;
	cout << "从后往前ch第一次出现的下标为:" << str1.rfind(ch) << endl;
}

        修改操作

        使用方式如下所示: 

void test_modify()
{
	string str1("hello");
	str1 += "world";
	cout << str1 << endl;
	str1.push_back('!'); //尾插单字符
	cout << str1 << endl;
	str1.append("haha");  //追加字符串
	cout << str1 << endl;

	string str2("hehe");
	str1.swap(str2); //交换
	cout << "str1 = " << str1 << endl;
	cout << "str2 = " << str2 << endl;
}

注意事项:

        一般我们在进行插入时,使用+=运算符会比较多,因为+=不仅可以用来插入字符,也可以用来插入字符串。

4.7 子串操作

        除此之外,string类还有两个较常用的接口,一个用于快速截取子串,一个用来获取C格式的字符串,如下所示:

        使用方式如下所示: 

void test_substr()
{
	string str1 = "hello world";
	string str2 = str1.substr(1, 8);  //截取从下标1处到下标8处的子串  
	cout << str2 << endl;
	cout << str2.c_str() << endl;  //获取C格式的字符串
}

4.8 非成员函数

        最后,除了上面定义在string类内的成员函数,C++还定义了一些全局函数string对象进行操作,典型的例子就是我们的流提取(>>)流插入(<<)运算符,这两个只能作为全局函数重载。具体的一些接口如下所示:

        使用方式如下所示: 

void test_general()
{
	string str1, str2;
	cout << "请输入一个字符串 :";
	cin >> str1;
	cout << "str1 = " << str1 << endl;
	cin.get();    //这里输入流中会剩下一个'\n',用get方法读取掉,否则getline遇到'\n'会直接停下来
	cout << "请输入一个字符串 :";
	getline(cin, str2);
	cout << "str2 = " << str2 << endl;

	if (str1 > str2)
		cout << "str1 > str2";
	else if(str1 < str2)
		cout << "str1 < str2";
	else 
		cout << "str1 == str2";
}


 以上,就是本期的全部内容啦🌸

制作不易,能否点个赞再走呢🙏


http://www.niftyadmin.cn/n/5154743.html

相关文章

测试服务器端口是否开通,计算退休时间

本案例知识点 netstat -tuln | grep 80 nestat 目前主机打开的网络服务端口&#xff0c;-tuln目前主机启动的服务&#xff0c;如图 报错说参数太多&#xff0c;仔细检查发现if后的中括号内&#xff0c;变量少双引号导致&#xff0c;改完之后运行显示22,25端口开放&#xff0…

动态规划算法实现0-1背包问题Java语言实现

问题介绍&#xff1a; 动态规划算法&#xff1a; 动态规划&#xff08;Dynamic Programming&#xff09;是一种解决多阶段决策问题的优化算法。它通过将问题分解为一系列子问题&#xff0c;并利用子问题的解来构建更大规模问题的解&#xff0c;从而实现对整个问题的求解。 动态…

inno setup 运行时进行文件复制和替换

问题描述&#xff1a; 当我们采用 inno setup进行打包时&#xff0c;需要实现将安装包中的某个文件进行替换&#xff0c;而且我们知道在Winodws系统可以有xcopy和copy两个命令可以提供该功能&#xff1b;而xcopy命令进行文件复制时会有如下提示&#xff1a; 此时需要手动输入字…

为什么有了MAC地址,还需要IP地址?

解释 搞懂这个问题&#xff0c;首先需要了解交换机的功能 交换机内部有一张MAC地址映射表&#xff0c;记录着MAC地址和端口的对应关系。 如果A要给B发送一个数据包&#xff0c;构造如下格式的数据结构&#xff1a; 到达交换机时&#xff0c;交换机内部通过自己维护的 MAC 地…

2023年11月5日网规考试备忘

早上题目回忆&#xff1a; pki体系 ipsec&#xff0c;交换安全&#xff08;流量抑制&#xff09; aohdlc bob metclaf —ethernet pon tcp三次握手 OSPF lsa&#xff1f;交换机组ospf配置问题&#xff0c;ping网关可通&#xff0c;AB不通 raid6 300G*8 网络利用率 停等协议10…

新技术前沿-2023-应用GPT提问模板写技术文章

参考一份万能的GPT提问模版&#xff01;直接套用&#xff01; 参考用GPT写技术文章是真爽&#xff01; 参考码住这篇 8200 字 ChatGPT 实战指南&#xff01;&#xff01; 1 GPT提问模板 想让GPT回答的内容符合我们所希望的&#xff0c;最最重要的一点就在于我们如何提问。提问…

基础Redis-Java客户端操作介绍

Java客户端操作介绍 2.基础-Redis的Java客户端a.介绍b.Jedisc.Jedis连接池d.SpringDataRedise.SpringDataRedis的序列化方式f.StringRedisTemplate 2.基础-Redis的Java客户端 a.介绍 Jedis 以Redis命令作为方法名称&#xff0c;学习成本低&#xff0c;简单实用。但是Jedis实例…

【手写数据库所需C语言基础】可变结构体,结构体成员计算,类型强制转换为统一类型,数据库中使用C语言方法和技巧

​专栏内容&#xff1a; 手写数据库toadb 本专栏主要介绍如何从零开发&#xff0c;开发的步骤&#xff0c;以及开发过程中的涉及的原理&#xff0c;遇到的问题等&#xff0c;让大家能跟上并且可以一起开发&#xff0c;让每个需要的人成为参与者。 本专栏会定期更新&#xff0c;…