【C++进阶(四)】STL大法--list深度剖析list迭代器问题探讨

💓博主CSDN主页:杭电码农-NEO💓

⏩专栏分类:C++从入门到精通⏪

🚚代码仓库:NEO的学习日记🚚

🌹关注我🫵带你学习C++
  🔝🔝


在这里插入图片描述

链表list

  • 1. 前言
  • 2. list的使用
    • 2.1 list的构造函数
    • 2.2 list迭代器的使用
    • 2.3 list容量相关操作
    • 2.4 list的增删查改
  • 3. list迭代器失效问题探讨
  • 4. 算法库函数和list的关系
    • 4.1 算法库函数的迭代器类型
    • 4.2 list不能使用的算法库函数
  • 5. 总结以及拓展

1. 前言

本质重点:

本章重点讲解list的接口函数的熟悉
并且讲解list迭代器失效的特性
最后讲解迭代器的功能分类以及
算法库函数中谁能用谁不能用

STL标准库中的list是一个
带头双向循环链表

和vector不同,list没有支持[ ]访问
以及resize和reserve容量相关的函数

这是因为list不能随机访问数据

并且list的迭代器的底层明显不是指针了
那它的底层到底是啥?
list会和vector一样有迭代器失效问题吗?

带着这些疑问,我们来进入今天的学习分享


2. list的使用

我们还是在网站:cplusplus中查询字典

在这里插入图片描述

和vector一样,list也有两个模板参数
但是第二个模板参数是和内存效率相关的
所以现在的学习暂时不用管它(它有缺省值)

list的使用分为以下几个阶段进行:

  • list的构造,析构,拷贝构造函数
  • list迭代器的使用
  • list容量相关的操作
  • list的增删查改

2.1 list的构造函数

list的构造函数:

在这里插入图片描述

第一个是无参构造,直接跳过
第二个是用n个val初始化list对象
第三个是用一段迭代器区间构造
第四个是用一个初始值构造

看起来平平无奇,来实操一下:

list<int> l1;//无参构造
list<int> l2(10,5);//用10个5初始化链表vector<int> vv{1,2,3,4,5,6};
list<int> l3(vv.begin(),vv.end());//用迭代器区间初始化list<char> l4('a');//用一个字符来初始化

list的拷贝构造和析构函数在使用
list时不会显示调用,所以将它们忽略掉


2.2 list迭代器的使用

虽然list的迭代器底层不是指针
但是可以把它理解为指针来使用

在这里插入图片描述
这四个函数都是老朋友了,不多介绍了

唯一值得注意的是下图:
在这里插入图片描述
begin和rend的指向相同
end和rbegin的指向相同

迭代器遍历链表:

vector<int> vv{1,3,5,7,9,11,13,15};
list<int> l(vv.begin(),vv.end());auto it = l.begin();
while(it!=l.end())
{cout << *it<< " ";it++;
}

范围for遍历链表:

vector<int> vv{1,3,5,7,9,11,13,15};
list<int> l(vv.begin(),vv.end());
for(auto x : l)
{cout << x<< " ";
}

注:list不支持随机访问,不能用[]

一个小tips:

在写用迭代器遍历容器时,我们往往
会写it!=l.end(),而不是it<l.end()
这是因为在list中,空间并不连续,用小于
符号很大概率会出错,但是在string
和vector中,空间是连续的,用<也没问题


2.3 list容量相关操作

这里最简单,和vector一模一样

在这里插入图片描述
在这里插入图片描述
只有这四个函数接口比较常用!


2.4 list的增删查改

首先映入眼帘的头插/头删,尾插/尾删

在这里插入图片描述

这几个函数的意思很明了,直接跳过

insert插入:

在这里插入图片描述

insert函数要输入一个迭代去位置进行插入
可以插入一个数据或插入一段迭代器区间

erase删除:

在这里插入图片描述
erase函数可以删除一个迭代器的数据
或者删除一段迭代器区间

clear清空有效元素:

在这里插入图片描述

list不像vector一样有size和capacity
list的clear函数是将链表清空
(除头节点外)

增删查改实操:

vector<int> vv{1,5,10,15,20,100,120};
list<int> ll(vv.begin(),vv.end());
auto pos = find(ll.begin(),ll.end(),20);
ll.insert(pos,250);
ll.erase(ll.begin()+2);
ll.clear();

3. list迭代器失效问题探讨

由于list的底层是双向带头循环链表
所以插入操作时,pos指向的节点的
位置始终都是一个位置,它们只改变
链接关系,并不连续,所以插入操作
并不会导致list迭代器失效

list迭代器失效只在erase删除时才会发生
erase删除的位置在空间上是唯一的
这个位置的数据被删除后,只是改变了
数据的链接关系,然而此位置已经不在
原先的链表中了,再次使用会出错!

STL库的erase提供了返回值来解决问题:

在这里插入图片描述
在这里插入图片描述
会返回被删除位置的下一个位置的迭代器
以后可以这样写代码来避免错误:

list<int> ll{1,2,3,4,5,6,7,8};
auto it=ll.begin();
while(it!=ll.end())
{if((*it)%2==0){it=erase(it);}else{it++;}
}

4. 算法库函数和list的关系

首先,迭代器从功能角度可以分类为:

  1. 正向迭代器: forward_iterator
  2. 双向迭代器: bidirectional_iterator
  3. 随机迭代器: random_iterator

它们分别支持且仅支持:

  1. 只支持++
  2. 支持++和- -
  3. 支持++,- -,+和-

常见的容器以及它们的迭代器类型:

容器迭代器功能
vector随机迭代器
list双向迭代器
stack不支持迭代器
queue不支持迭代器
deque随机迭代器
set / multiset双向迭代器
map / multimap双向迭代器
priority_queue不支持迭代器

4.1 算法库函数的迭代器类型

现在再来看算法库的sort函数:

在这里插入图片描述

它的函数模板支持的是随机迭代器

再来看看算法库的reverse函数:

在这里插入图片描述

它的函数模板支持的是双向迭代器

我先给出以下的结论:

  1. 若函数模板给的随机迭代器
    则只能传有随机迭代器的容器

  2. 若函数模板给的双向迭代器
    则可以传有随机或者双向迭代器的容器

  3. 若函数模板给的单向迭代器
    则三种迭代器都可以传进来!

可以看出它支持向上兼容!


4.2 list不能使用的算法库函数

可见,list是双向迭代器,而sort函数的
函数模板是随机迭代器
所以库函数中的sort无法排序链表

但是仔细查阅字典可以发现:
list类自己实现了一个不同于算法库的排序

在这里插入图片描述

此排序专门用于排序list中的数据!

虽然list有自己的sort函数来排序
但是当数据很多时,list自我实现的
sort的效率低的惊人!甚至还不如
将list的数据导入vector容器
再在vector容器中使用算法库的排序
排完序再导入到list中,这一步骤快!


5. 总结以及拓展

总的来说list的使用和vector大同小异
当然,要掌握list,list的模拟实现是不可少的
list的模拟实现将在下一节分享给大家

本章笔记:

  1. 请自行熟悉list接口函数

  2. list的insert不会导致迭代器失效
    而erase会致使迭代器失效

  3. 迭代器从功能上可以划分为三种
    它们向上兼容彼此

  4. list不能使用算法库中的sort
    但list类中实现了专门排序链表的函数

拓展阅读:

迭代器相关拓展知识


🔎 下期预告:list的模拟实现以及对比vector的区别 🔍

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

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

相关文章

通过 Keycloak 结合 OAuth2.0协议进行 Amazon API Gateway 鉴权

1. 简介 本文介绍了如何通过 Keycloak&#xff0c;并结合 Amazon API Gateway 内置的授权功能&#xff0c;完成对 Amazon 资源请求的鉴权过程。API Gateway 帮助开发者安全的的创建、发布、维护并管理 API 的访问。在中国区&#xff0c;由于Cognito 仍未上线&#xff0c;因此使…

Groovy 下载安装

Groovy 简介 在某种程度上&#xff0c;Groovy 可以被视为Java 的一种脚本化改良版,Groovy 也是运行在 JVM 上&#xff0c;它可以很好地与 Java 代码及其相关库进行交互操作。它是一种成熟的面向对象编程语言&#xff0c;既可以面向对象编程&#xff0c;又可以用作纯粹的脚本语…

【python零基础入门学习】python基础篇之文件对象open、模块以及函数的使用(三)

本站以分享各种运维经验和运维所需要的技能为主 《python》&#xff1a;python零基础入门学习 《shell》&#xff1a;shell学习 《terraform》持续更新中&#xff1a;terraform_Aws学习零基础入门到最佳实战 《k8》暂未更新 《docker学习》暂未更新 《ceph学习》ceph日常问题解…

leetcode 922. 按奇偶排序数组 II

2023.9.4 本题较为简单&#xff0c;构造一个和nums相同大小的数组ans&#xff0c;然后遍历判断nums中的元素&#xff0c;若为奇数则放在ans中的奇数索引位置&#xff0c;偶数则放在ans中的偶数索引位置。 代码如下&#xff1a; class Solution { public:vector<int> sor…

智能安全帽~生命体征检测与危险气体检测一体化集成设计还是蓝牙无线外挂式方式好?

生命体征&#xff08;心率、血氧等&#xff09;检测&上报平台&#xff0c;危险气体采集&上报平台&#xff0c;是智能安全帽产品中常见的两种选配件&#xff0c;它们的实现有两种典型的模式&#xff1a; 1&#xff09;将传感器集成到主板上&#xff0c;做成一体化的智能…

三维模型OBJ格式轻量化的数据压缩与性能平衡分析

三维模型OBJ格式轻量化的数据压缩与性能平衡分析 三维模型的OBJ格式轻量化数据压缩在保持性能的同时&#xff0c;可以减小文件大小、提高加载速度和节省存储空间。然而&#xff0c;在进行数据压缩时&#xff0c;需要权衡压缩比率和模型质量之间的关系&#xff0c;并考虑不同应用…

AIGC+思维导图:提升你的学习与工作效率的「神器」

目录 一、产品简介 二、功能介绍 2.1 AI一句话生成思维导图 2.2百万模版免费用 2.3分屏视图&#xff0c;一屏读写 2.4团队空间&#xff0c;多人协作 2.5 云端跨平台化 2.6 免费够用&#xff0c;会员功能更强大 2.7 支持多种格式的导入导出 三、使用教程 3.1 使用AI…

【Linux】高级IO和多路转接 | select/poll/epoll

多路转接和高级IO 咳咳&#xff0c;写的时候出了点问题&#xff0c;标点符号全乱了&#xff08;批量替换了几次&#xff09;&#xff0c;干脆就把全文的逗号和句号都改成英文的了&#xff08;不然代码块里面的代码都是中文标点就跑不动了&#xff09; 1.高级IO 1.1 五种IO模型…

Android逆向学习(番外一)smali2java部分文件无法反编译的bug与修复方法

Android逆向学习&#xff08;番外一&#xff09;smali2java部分文件无法反编译的bug与修复方法 一、前言 昨天我和往常一样准备着android逆向&#xff08;四&#xff09;的博客&#xff0c;结果发现smali2java对某些文件无法进行逆向&#xff0c;我不知道windows会不会产生这…

【数据结构初阶】一. 复杂度讲解

相关代码gitee自取&#xff1a; C语言学习日记: 加油努力 (gitee.com) 接上期&#xff1a; 学C的第三十四天【程序环境和预处理】_高高的胖子的博客-CSDN博客 1 . 算法效率 &#xff08;1&#xff09;. 什么是数据结构&#xff1a; 数据结构(Data Structure)是计算机存储、…

联发科MTK6762/MT6762核心板_安卓主板小尺寸低功耗4G智能模块

MT6762安卓核心板是一款基于MTK平台的高性能智能模块&#xff0c;是一款工业级的产品。该芯片也被称为Helio P22。这款芯片内置了Arm Cortex-A53 CPU&#xff0c;最高可运行于2.0GHz。同时&#xff0c;它还提供灵活的LPDDR3/LPDDR4x内存控制器&#xff0c;此外&#xff0c;Medi…

什么是专线网络?互联网专线为什么贵

宽带主要分为专线和家庭宽带。一般来说&#xff0c;企业服务器和云服务提供商通常使用专线宽带&#xff0c;普通家庭接入家庭宽带。 什么是专线网络&#xff1f; 专线是指依托国内骨干网络和宽带城市网络资源&#xff0c;为企业集团客户提供专线接入方式互联网城市互联网&…

Github 下载指定文件夹(git sparse-checkout)

比如要下载这里的 data_utils 步骤 1、新建空文件夹&#xff0c;并进入新建的空文件夹。 2、git init 初始化 3、git remote add origin 添加远程仓库 4、git config core.sparsecheckout true 允许稀疏检出 5、git sparse-checkout set 设置需要拉取的文件夹&#xff08;可…

如何通过Pytest的插件,轻松切换自动化测试的环境地址?

前言 前面小编介绍了如何通过Pytest的插件来实现自动化测试的环境的切换&#xff0c;当时使用的方法是通过钩子函数进行获取命令行参数值&#xff0c;然后通过提前配置好的参数进行切换测试环境地址。加我VX&#xff1a;atstudy-js 回复“测试”&#xff0c;进入 自动化测试学…

浅谈一下酒吧和酒馆的不同

相信有很多朋友还不怎么清楚酒吧和酒馆的区别是什么&#xff0c;这里为大家简单介绍一下两者的不同&#xff0c;个人见解&#xff0c;如有错漏&#xff0c;欢迎指出。一、首先是他们的经营范围不同酒馆经营通常包含酒水和餐饮&#xff0c;适合朋友聚会或者是和商业伙伴聊天。而…

14.Redis 主从复制

Redis 主从复制 redis 主从复制配置 redis 主从复制启动 redis 主从复制断开 redis 主从复制主从复制构特点主从复制的拓扑结构一主一从⼀主多从树状主从 主从复制原理数据同步psync 运行流程全量复制流程部分复制流程实时复制 关于从节点何时晋升成主节点总结 redis 主从复制 …

操作系统(OS)与系统进程

操作系统&#xff08;OS&#xff09;与系统进程 冯诺依曼体系结构操作系统(Operator System)进程基本概念进程的描述&#xff08;PCB&#xff09;查看进程通过系统调用获取进程标示符&#xff08;PID&#xff09;通过系统调用创建进程&#xff08;fork&#xff09;进程状态&…

【Linux】线程池

一、概念 1.线程池是一种利用池化技术思想来实现的线程管理技术&#xff0c;主要是为了复用线程、便利地管理线程和任务、并将线程的创建和任务的执行解耦开来。我们可以创建线程池来复用已经创建的线程来降低频繁创建和销毁线程所带来的资源消耗。 2.线程池是一种线程使用模式…

OpenCV:实现图像的负片

负片 负片是摄影中会经常接触到的一个词语&#xff0c;在最早的胶卷照片冲印中是指经曝光和显影加工后得到的影像。负片操作在很多图像处理软件中也叫反色&#xff0c;其明暗与原图像相反&#xff0c;其色彩则为原图像的补色。例如&#xff0c;颜色值A与颜色值B互为补色&#…

陶氏公司将出席2023第二届中国汽车碳中和峰会

2023第二届中国汽车碳中和峰会将于10月19日-20日在上海举办。 本次峰会将为行业领导者、政策制定者和专家提供一个平台&#xff0c;讨论汽车行业减少碳排放的策略。专家们将从政策、供应链、ESG、替代能源解决方案、汽车材料创新、法律等不同领域分享碳中和与可持续策略。 通…