STL容器搜索:当直接访问STL容器时,如何执行有效和正确的搜索?

掌握STL容器搜索技巧:在C++中实现高效和准确的数据访问

  • 一、简介
  • 二、std::vector, std::deque, std::list
  • 三、std::map, std::multimap, std::set, std::multiset
  • 四、std::string
  • 六、总结

一、简介

本文主要了解如何在直接访问c++容器时高效地进行搜索。在STL容器中搜索,要牢记一个原则:如果可以的话,最好使用容器方法来搜索而不是使用外部算法接口。

有三个原因:

  • 它更快:在排序的容器中,所有方法都受益于排序集合中的快速对数搜索。此外,std::string方法实现了最优算法,并受益于字符串的内部表示。
  • 它更自然:std::mapstd::multimap方法可以直接搜索键,而不像算法那样必须查找std::pair< key, Value>,因为它们的迭代器可以直接指向。
  • 它在某些情况下更正确:在排序容器(如mapset)中,所有方法都使用等价而不是相等,而某些算法(如std::countstd::find使用相等)则不是这样。

现在让我们通过研究如何将它应用到 STL 提供的各种容器来深入了解更多细节。

二、std::vector, std::deque, std::list

这些容器没有公开任何与搜索相关的方法,只能通过算法来搜索。

三、std::map, std::multimap, std::set, std::multiset

这些容器有5个类方法,它们与一些算法共享它们的名称:count, find, equal_range, lower_boundupper_bound。在上一篇文章中有讲解过这些算法。

这些方法和算法的比较:

容器方法比算法更正确?比算法还快?比算法更自然?
count
find
equal_range相同
lower_bound相同
upper_bound相同
  • 更好的正确性是因为使用了等效而不是相等。
  • 更好的性能来自于为序列容器对元素进行排序这一事实。对于关联容器来说,这是因为它们的迭代器不是随机访问的,所以算法不能通过直接跳过所需的元素来执行分割(它们必须从头开始并向上移动到它们的位置),而容器的内部没有这种约束。
  • 它们对于映射来说更自然,因为传递给各种方法的参数是一个键,而不是std::pair<key, Value>

注意,没有与std::binary_search等价的容器方法。检查容器中是否存在一个键:

  • 对于std::mapstd::set:比较find的结果与end迭代器的结果。或者使用count方法。count作为方法不会引起任何性能问题,因为,像find一样,它在第一个与搜索的键相等的键处停止(因为根据std::mapstd::set的定义,只能有一个键与搜索的键相等)。
  • 对于std::multimapstd::multiset:因为count不会在第一个与搜索的键相等的键处停止,所以find在这里比count有优势。

std::multimapstd::multiset中,find方法返回与搜索值相等的任何元素,而不一定是第一个元素。如果确实需要第一个元素,可以使用equal_range,因为它具有简单的接口;或者,如果觉得equal_range太慢(因为它显示了整个范围),而只是需要第一个元素,那么可以使用lower_bound。但是,lower_bound同样也有它的缺点,也必须付出代价。

四、std::string

string实际上有24个搜索方法。它们被分成6组,每组有4个重载。对于所有组,4个重载的形式为:

  • 搜索由std::string给出的字符串。
  • 搜索由char*和size给出的字符串。
  • 搜索由char*给出的字符串(止于null字符)。
  • 搜索一个字符。

并且所有4个重载都以搜索字符串中的起始位置作为参数,默认值为0(从字符串的开头开始搜索)。

以下是6组方法:

  1. find 函数:用于从字符串中查找给定子字符串 str 的第一个匹配项。它返回子字符串在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 0。

    size_t find(const std::string& str, size_t pos = 0) const;
    
  2. rfind 函数:与 find 函数类似,但它从字符串的末尾开始向前搜索子字符串 str 的最后一个匹配项。它返回子字符串在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 std::string::npos,表示从末尾开始搜索。

    size_t rfind(const std::string& str, size_t pos = npos) const;
    
  3. find_first_of 函数:用于从字符串中查找与给定字符串 str 中的任何字符匹配的第一个字符。它返回找到的字符在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 0。

    size_t find_first_of(const std::string& str, size_t pos = 0) const;
    
  4. find_last_of 函数:与 find_first_of 函数类似,但它从字符串的末尾开始向前搜索与给定字符串 str 中的任何字符匹配的最后一个字符。它返回找到的字符在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 std::string::npos,表示从末尾开始搜索。

    size_t find_last_of(const std::string& str, size_t pos = npos) const;
    
  5. find_first_not_of 函数:用于从字符串中查找第一个不在给定字符串 str 中的字符。它返回找到的字符在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 0。

    size_t find_first_not_of(const std::string& str, size_t pos = 0) const;
    
  6. find_last_not_of 函数:与 find_first_not_of 函数类似,但它从字符串的末尾开始向前搜索第一个不在给定字符串 str 中的字符。它返回找到的字符在当前字符串中的位置索引,如果找不到则返回 std::string::npospos 参数是可选的,用于指定搜索的起始位置,默认为 std::string::npos,表示从末尾开始搜索。

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

在线性时间内实现字符串算法并不容易。string方法以最佳方式实现它们,当在字符串中搜索某些内容时,可以直接使用容器的方法。

六、总结

本文系统地介绍了在直接访问STL容器时执行有效和正确搜索的方法。通过理解STL容器的内部机制和使用适当的搜索技巧,可以提高代码的性能和可读性。关键要点包括使用迭代器和成员函数进行搜索,利用算法库提供的函数进行查找,以及根据不同的容器类型选择最佳搜索方法。通过掌握本文提供的技术和原则,读者将能够在使用STL容器时更加自信地进行搜索操作,提高代码的质量和效率。

在这里插入图片描述

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

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

相关文章

5_vscode+valgrind+gdb调试程序

需求 项目程序, 读取串口数据, 出现程序崩溃问题valgrind 可以调试定位内存问题: 内存泄漏,非法地址访问,越界访问等内存问题vscode gdb 可视化调试效果, 比命令行简单快捷很多期望使用vscode valgrind gdb 调试程序内存异常, 崩溃退出的问题 环境准备 sudo apt install v…

【个人博客搭建】(5)Sqlsugar实体创建数据库数据

1、在appsettings.json文件中配置SqlServer数据库连接字符串信息。&#xff08;后续考虑添加MySQL数据库等类型&#xff09; "DBS": [/*对应下边的 DBTypeMySql 0,SqlServer 1,*/{"ConnId": "plateau.poetize.2024","DBType": 1,&qu…

书生·浦语大模型实战营之XTuner多模态训练与测试

书生浦语大模型实战营之XTuner多模态训练与测试 目录 XTuner多模态训练与测试给LLM装上电子眼:多模态LLM原理简介文本单模态文本+图像多模态电子眼:LLaVA方案简介LLaVA训练阶段示意图LLaVA测试阶段示意图项目实践环境准备XTuner安装概述Pretrain阶段Finetune阶段训练数据构建…

14.基础乐理-音级、基本音级、变化音级

音级&#xff1a; 乐音体系中的每一个音&#xff0c;都叫 音级。 基本音级&#xff1a; 基本音级是 CDEFGAB 它们七个&#xff0c;在钢琴上使用白键展示的&#xff0c;没有任何升降号、没有任何重升重降号的。 变化音级&#xff1a; 除了 CDEFGAB 这七个音&#xff0c;都叫变化…

vue 下载文件 处理后台返回的文件流

1. 下载文件很常见&#xff0c;下载成各种格式的也很常见&#xff0c;本质就是后台返回一个文件流&#xff0c;我们前端去处理一下就行&#xff0c;但是如果因为某些条件&#xff0c;没有返回文件流&#xff0c;返回告诉你&#xff0c;文件出现错误了&#xff0c;那我们就需要把…

C#中对任务和线程并行运行,测试其执行时间

为了提高程序的的性能&#xff0c;减少流程的时间时&#xff0c;我们总是会考虑将流程分成多个部分&#xff0c;并在同一时刻执行它。这就是并行性&#xff0c;那么在并行中线程和任务哪个更好呢&#xff1f; 1.使用线程 在此示例中&#xff0c;我们将每个线程添加到一个列表…

跑步新潮流:Shokz韶音OpenRun Pro配色引领风骚

Shokz韶音OpenRun Pro自推出以来,凭借其卓越的产品性能,赢得了广泛运动爱好者的青睐。尤其是,新近推出的特别配色版本,包括“牛仔蓝”和“水泥灰活力橙”的基普乔格联名款,这些全新色调进一步突显了使用者的个性化需求。 在众多选择中,Shokz韶音OpenRun Pro的“牛仔蓝”配色受…

neo4j的Cypher的语法记录

1.参考1 Neo4j图数据库及Cypher语法基础 | Quantum Bit 参考2 常用CYPHER查询&#xff08;二&#xff09; - Yc-Ma Blog 1.1 查询 MATCH (n) RETURN n LIMIT 25 1.2 删除节点及关系 删除节点 删除关系 MATCH (n:PersonWeb {personWebName: "玉笙lalala"}) D…

Makefile后知后觉几件事

导言 以前经常用Makefile&#xff0c;但是仅是用而已&#xff0c;没有从头折腾过&#xff0c;所以&#xff0c;谈不上很深的理解。 最近针对一些场景做了一些实践&#xff0c;对于Makefile中经常遇到的各种变量类型&#xff0c;和目标依赖关系&#xff0c;以及与make并行编译…

使用 Flask 和 Flask-Login 构建用户认证的 Web 应用程序

在本篇技术博客中&#xff0c;我们将学习如何使用 Flask 框架和 Flask-Login 扩展构建一个具有用户认证功能的简单 Web 应用程序。我们将从创建 Flask 应用实例开始&#xff0c;然后逐步添加用户认证功能。 1. 安装依赖库 首先&#xff0c;确保您已经安装了 Flask、Flask-PyM…

create (MUs)方法

创建移动对象 (create (MUs)) 此部分来自 Plant Simulation 的对象参考指南,介绍用于创建移动对象的 create (MUs) 方法。 功能: 创建指定类型的移动对象 (MU) 实例。语法: <MU-路径>.create(MU-位置:any[, 位置:长度, 复制统计数据:布尔值, 前驱编号:整型]) → 对象…

学习前端第二十五天(构造器和操作符‘new’,可选链‘?.’,symbol类型)

一、构造器和操作符‘new’ 构造函数&#xff1a; 命名以大写字母开头。只能由 "new" 操作符来执行。 当一个函数被使用 new 操作符执行时&#xff0c;它按照以下步骤&#xff1a; 一个新的空对象被创建并分配给 this。函数体执行。通常它会修改 this&#xff0c…

Spring Boot 学习(7)——条件注解

看到今天&#xff0c;方才把前面 demo2 中的一些地方想明白。就好比一个数学系的研究生在给初中生讲数据题的时候&#xff0c;不知不觉就会用到超纲的知识&#xff0c;这也是为什么各类大神的教程&#xff08;视频或文字&#xff09;按些步骤来总有对不上的地方&#xff0c;因为…

面试Spring框架

什么是Spring框架&#xff1f; Spring框架是一个开源的Java应用程序框架&#xff0c;提供了综合的基础设施支持&#xff0c;用于开发Java企业应用程序。它涵盖了从基本的核心容器到全面的企业服务&#xff0c;可以用于构建任何规模的应用程序。 Spring框架的核心特性是什么&am…

JavaSE高阶篇-反射

第一部分、Junit单元测试 1&#xff09;介绍 1.概述:Junit是一个单元测试框架,在一定程度上可以代替main方法,可以单独去执行一个方法,测试该方法是否能跑通,但是Junit是第三方工具,所以使用之前需要导入jar包 2&#xff09;Junit的基本使用&#xff08;重点啊&#xff09; 1.…

C++中的运算符

一、算数运算符 1.1 加减乘除取模 #include <iostream> using namespace std;int main() {//加减乘除int a1 10;int b1 5;cout << "a1 b1 " << a1 b1 << endl;cout << "a1 - b1 " << a1 - b1 << endl;co…

周期规律的应用(上)——双因子定价模型的应用:择时与配置

周期规律的应用&#xff08;上&#xff09;——双因子定价模型的应用&#xff1a;择时与配置 原创 林晓明&#xff0c;源洁莹 华泰证券金融工程 2024-04-20 09:50 广东 本次推送是华泰金工团队《周期论道&#xff1a;洞察经济金融系统的节奏与脉动》会议论坛的纪要&#xff0c…

QT c++ QWaitCondition 线程等待条件

本文描述&#xff0c;QT c QWaitCondition 线程等待条件&#xff0c;在文章后面列出了参考代码下载链接。 如果不用它&#xff0c;可能需要通过定时循环检查某个变量的值满足条件&#xff0c;执行相应的代码&#xff0c;程序空跑&#xff0c;占用CPU资源。 而且定时循环并不…

IP代理池纯净度有什么重要性?

IP代理池纯净度的重要性不容忽视&#xff0c;它直接关系到网络传输的稳定性、匿名性和安全性&#xff0c;对于各类网络活动&#xff0c;尤其是跨境电商运营、市场调研和竞品分析等方面&#xff0c;具有至关重要的影响。 第一点&#xff0c;纯净的IP代理池能够保障用户隐私&…

跟着Carl大佬学leetcode之844 比较含退格的字符串

来点强调&#xff0c;刷题是按照代码随想录的顺序进行的&#xff0c;链接如下https://www.programmercarl.com/本系列是记录一些刷题心得和学习过程&#xff0c;就看到题目自己先上手试试&#xff0c;然后看程序员Carl大佬的解释&#xff0c;自己再敲一遍修修补补&#xff0c;练…