【C++】C++中的list

一、介绍

       官方给的 list的文档介绍

简单来说就是:

        list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。list的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。list与forward_list非常相似:最主要的不同在于forward_list是单链表,只能朝前迭代,已让其更简单高效。与其他的序列式容器相比(array,vector,deque),list通常在任意位置进行插入、移除元素的执行效率更好。与其他序列式容器相比,list和forward_list最大的缺陷是不支持任意位置的随机访问,比如:要访问list的第6个元素,必须从已知的位置(比如头部或者尾部)迭代到该位置,在这段位置上迭代需要线性的时间开销;list还需要一些额外的空间,以保存每个节点的相关联信息(对于存储类型较小元素的大list来说这可能是一个重要的因素。

这个时候大家可能觉得,都是有序列表,那么和vector有什么区别和对比吗?实际上和我们学习数据结构时对链表和数组的对比很像,我来介绍一下:
 

std::list

  • std::list 是一个双向链表,支持在常数时间内对序列的任何位置进行插入和删除操作。
  • 由于其链表的性质,list 不支持快速随机访问,即不能通过索引以常数时间访问元素(例如 list[5] 是非法的)。
  • list 更适用于元素频繁插入和删除的场景,尤其是在序列的头部和尾部,或者你不需要通过索引来访问元素。
  • 迭代器失效问题较少,插入和删除操作不会导致除了被操作的元素之外的迭代器失效。
  • 在内存中不是连续存储的,因此不支持指针算术运算,并且可能导致较差的缓存性能。

std::vector

  • std::vector 是一个动态数组,可以在末尾快速地添加或移除元素(均摊常数时间复杂度),而且支持快速随机访问,即可以以常数时间访问任意位置的元素。
  • vector 的中间或开头插入或删除元素可能会导致较高的性能开销,因为这些操作需要移动插入点之后(或删除点之后)的所有元素。
  • 适用于需要经常随机访问元素,但对于插入和删除的频率较低的场景。
  • 在内存中是连续存储的,这意味着可以使用指针算术,并且有助于优化缓存使用。
  • vector 重新分配更大的内存空间以容纳更多元素时,所有的迭代器、引用和指针都可能失效。

那么list到底长什么样子呢?上图片,是不是就好理解了

二、list的使用

        作为STL(标准模板库)中的一个类,我们这篇blog的任务就是学习其的使用。

构造函数

构造函数接口说明
list()构造空的list
list (size_type n, const value_type& val = value_type())构造的list中包含n个值为val的元素
list (const list& x)拷贝构造函数
list (InputIterator first, InputIterator last)[first, last)区间中的元素构造list

上面虽然用了不少代名词,我们直接上代码例子分析自然就清楚了,分析在代码中。

(如果迭代器看不懂可以看这一篇【C++】C++中的vector-CSDN博客,里面详细介绍了)

#include <iostream>
#include <list>
using namespace std;
int main()
{//list<int> l1; // 构造空的l1list<int> l2(4, 100); // l2中放4个值为100的元素list<int> l3(l2.begin(), l2.end()); // 用l2的[begin(), end())左闭右开的区间构造l3list<int> l4(l3); // 用l3拷贝构造l4// 以数组为迭代器区间构造l5int array[] = { 16,2,77,29 };std::list<int> l5(array, array + sizeof(array) / sizeof(int));// 用迭代器方式打印l5中的元素for (std::list<int>::iterator it = l5.begin(); it != l5.end(); it++)std::cout << *it << " ";std::cout << endl;// C++11范围for的方式遍历for (auto& e : l5)std::cout << e << " ";std::cout << endl;return 0;
}

其实我们可以看出来,list这个类和之前的使用类的方法是基本一致的,不过他需要一个<int>来确定这个序列容器的类型,比如int,char....,就是list<int>可以当成一个整体,和vector很像。

list iterator的使用

         此处,大家可暂时将迭代器理解成一个指针,该指针指向list中的某个节点
函数声明

接口说明

begin end

获取第一个数据位置的iterator/const_iterator, 获取最后一个数据的下一个位置 的iterator/const_iterator

rbegin + rend

获取最后一个数据位置的reverse_iterator,获取第一个数据前一个位置的 reverse_iterator

 PS:

1. begin end 为正向迭代器,对迭代器执行 ++ 操作,迭代器向后移动
2. rbegin(end) rend(begin) 为反向迭代器,对迭代器执行 ++ 操作,迭代器向前移动
上代码:
#include <iostream>
#include <list>
using namespace std;void print_list(const list<int>& l)
{// 注意这里调用的是list的 begin() const,返回list的const_iterator对象// 保护数据通过将l声明为常量引用,我们保证了在print_list函数内部无法修改列表l的内容。// 这意味着无法添加、删除或修改列表中的任何元素。这是一种良好的编程实践,// 特别是当函数的目的仅仅是读取数据而不修改数据时。for (list<int>::const_iterator it = l.begin(); it != l.end(); ++it){cout << *it << " ";//如果不同const 就通过//*it = 10; 编译不通过}cout << endl;
}
int main()
{int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };list<int> l(array, array + sizeof(array) / sizeof(array[0]));// 使用正向迭代器正向list中的元素for (list<int>::iterator it = l.begin(); it != l.end(); ++it)cout << *it << " ";cout << endl;// 使用反向迭代器逆向打印list中的元素for (list<int>::reverse_iterator it = l.rbegin(); it != l.rend(); ++it)cout << *it << " ";cout << endl;return 0;
}

常用的成员方法

list capacity

函数声明

接口说明

empty
检测 list 是否为空,是返回 true ,否则返回 false
size
返回 list 中有效节点的个数

列表元素访问

函数声明

接口说明

empty
检测 list 是否为空,是返回 true ,否则返回 false
size
返回 list 中有效节点的个数

 list modifiers

函数声明
接口说明
push_frontlist首元素前插入值为val的元素
pop_front删除list中第一个元素
push_backlist尾部插入值为val的元素
pop_back删除list中最后一个元素
insertlist position 位置中插入值为val的元素
erase 删除list position位置的元素
swap交换两个list中的元素
clear清空list中的有效元素
上代码看实现:
#include <iostream>
#include <list>
#include <vector>
using namespace std;
void PrintList(list<int>& l)
{for (auto& e : l)cout << e << " ";cout << endl;
}
//===============================================================
// push_back/pop_back/push_front/pop_front
void TestList1()
{cout << "TestList1()" << endl;int array[] = { 1, 2, 3 };list<int> L(array, array + sizeof(array) / sizeof(array[0]));// 在list的尾部插入4,头部插入0L.push_back(4);L.push_front(0);PrintList(L);// 删除list尾部节点和头部节点L.pop_back();L.pop_front();PrintList(L);
}
//================================================================
// insert /erase 
void TestList2()
{cout << "TestList2()" << endl;int array1[] = { 1, 2, 3 };list<int> L(array1, array1 + sizeof(array1) / sizeof(array1[0]));// 获取链表中第二个节点auto pos = ++L.begin();cout << *pos << endl;// 在pos前插入值为4的元素L.insert(pos, 4);PrintList(L);// 在pos前插入5个值为5的元素L.insert(pos, 5, 5);PrintList(L);// 在pos前插入[v.begin(), v.end)区间中的元素vector<int> v {7, 8, 9 };L.insert(pos, v.begin(), v.end());PrintList(L);// 删除pos位置上的元素L.erase(pos);PrintList(L);// 删除list中[begin, end)区间中的元素,即删除list中的所有元素L.erase(L.begin(), L.end());PrintList(L);
}
// resize/swap/clear
void TestList3()
{cout << "TestList3()" << endl;// 用数组来构造listint array1[] = { 1, 2, 3 };list<int> l1(array1, array1 + sizeof(array1) / sizeof(array1[0]));PrintList(l1);// 交换l1和l2中的元素list<int> l2;l1.swap(l2);PrintList(l1);PrintList(l2);
//使用resize将l2的大小先增加到5个元素,所有新添加的元素都将被赋值为99l2.resize(5, 99);PrintList(l2);// 将l2中的元素清空l2.clear();cout << l2.size() << endl;
}int main()
{TestList1();TestList2();TestList3();return 0;
}

        好了,目前通过上面这一段精简的代码,我们把常用的成员方法基本解决了,但是list的成员方法实在太多,很多操作都是很特殊,不常见的,但是如果刚好需要又非常方便,所以就是可以在需要的时候查官方文档。

list的迭代器失效

在之前我们学习过vector的迭代器会有失效的情况,原因很简单,指针失效了,那么list会不会有这种情况呢?答案是有的,前面说过,此处大家可将迭代器暂时理解成类似于指针,迭代器失效即迭代器所指向的节点的无效,即该节点被删除了。因为list的底层结构为带头结点的双向循环链表,因此在list中进行插入时是不会导致list的迭代器失效的,只有在删除时才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不会受到影响。所以影响相对vector来说比较小。

理解了吗?两段代码来检测一下大家

#include <iostream>
#include <list>
using namespace std;void TestListIterator1()
{int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };list<int> l(array, array + sizeof(array) / sizeof(array[0]));auto it = l.begin();while (it != l.end()){l.erase(it);++it;}
}void TestListIterator2()
{int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };list<int> l(array, array + sizeof(array) / sizeof(array[0]));auto it = l.begin();while (it != l.end()){l.erase(it++); // it = l.erase(it);}
}int main()
{TestListIterator1();TestListIterator2();return 0;
}

是 TestListIterator1 出错 还是 TestListIterator2 出错?

如果你一眼就看出了,那么恭喜你,你掌握了。

其实很简单,第一个当调用erase(it)后,it被删除,使得it失效。尝试在失效的迭代器上进行操作(比如递增++it)是未定义行为。

      第二个,l.erase(it++):这里使用了“后置递增”运算符,它创建了it的一个副本,然后将副本传递给erase方法。erase删除了当前迭代器指向的元素,然后it被递增,指向下一个元素。因为it在递增前已经复制给erase,所以即使在删除当前元素后,递增操作是在一个新的、未被修改的迭代器上进行的,这保证了迭代器的有效性。或者可以这样写,等价的it = l.erase(it);erase函数返回下一个有效的迭代器,然后将其赋值给it。这样,it始终保持有效,且指向当前元素的下一个元素。

三、结语

到此为止,我们已经把list的基本使用方法学习结束了,list的成员方法十分丰富,这篇文章就是介绍了常用的,让大家基本会使用,目前你也可以用这种双向列表来实现一些复杂的算法,我在下面了可以给大家写一个。等我有时间再出一篇,模拟实现list的blog,理解他的底层实现,有缘再见,朋友!

实现的经典算法

约瑟夫环问题(Josephus Problem)。这个问题的一个版本可以描述如下:N个人围成一圈,从第一个人开始报数,每报到M时,该人被淘汰,接着从下一个人开始继续报数,直到所有人都被淘汰。任务是按顺序输出被淘汰人的编号。

#include <iostream>
#include <list>
using namespace std;void JosephusProblem(int N, int M) {// 初始化人员列表,编号从1到Nlist<int> people;for (int i = 1; i <= N; ++i) {people.push_back(i);}auto it = people.begin(); // 迭代器指向第一个人while (!people.empty()) {// 模拟报数,M-1次移动迭代器(因为从当前人开始报数)for (int count = 1; count < M; ++count) {++it;// 如果迭代器超过了末尾,重新从头开始if (it == people.end()) {it = people.begin();}}// 报到M,移除当前人,并输出编号cout << *it << " ";it = people.erase(it); // erase返回下一个元素的迭代器// 如果列表不为空,但迭代器已经到达末尾,需要重新指向开头if (it == people.end() && !people.empty()) {it = people.begin();}}cout << endl;
}int main() {int N = 7; // 人数int M = 3; // 报数淘汰JosephusProblem(N, M);return 0;
}

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

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

相关文章

DashOJ-8.奇偶统计

题目链接&#xff1a; 题目详情 - 奇偶统计 - DashOJ 思路&#xff1a; &#xff08;while循环加if分支语句&#xff09; 巧用死循环 while(1) 然后在里面第一句就判断输入的数字是否等于0 if(x0) &#xff0c;如果 等于0就直接break跳出循环 或者用 while(cin>>x) 代…

后端SpringBoot+Mybatis 查询订单数据表奇怪报错加一

排错过程&#xff1a; 看报错意思是SQL语句存在错误&#xff0c;然后使用图形化工具运行这个SQL语句 其实这里稍微细心想一下就能发现问题&#xff0c;但是当时没深入想&#xff0c;就觉得order表前加了数据库名字影响不大&#xff0c;所以感觉SQL语句是没问题的&#xff0c;然…

STM32八种I/O口模式

STM32八种I/O口模式 文章目录 STM32八种I/O口模式前言一、stm32八种I/O类型二、区别1.模拟输入2.浮空输入3.上拉输入4.下拉输入5.推挽输出6.开漏输出7.复用推挽输出8.复用推挽输出 总结 前言 作为两年嵌入式软件攻城狮&#xff0c;还没仔细去理解过STM32的GPIO的八种使用模式&…

力扣爆刷第111天之CodeTop100五连刷41-45

力扣爆刷第111天之CodeTop100五连刷41-45 文章目录 力扣爆刷第111天之CodeTop100五连刷41-45一、232. 用栈实现队列二、4. 寻找两个正序数组的中位数三、31. 下一个排列四、69. x 的平方根五、8. 字符串转换整数 (atoi) 一、232. 用栈实现队列 题目链接&#xff1a;https://le…

引脚数量最少的单片机

引脚数量最少的单片机 2款SOT23-6封装单片机介绍 参考价格 PMS150C-U06 整盘单价&#xff1a;0.19688&#xff0c;该芯片为中国台湾品牌PADAUK(应广) SQ013L-SOT23-6-TR 整盘单价&#xff1a;0.27876&#xff0c;该芯片为国产&#xff1a;holychip(芯圣电子) 上述价格为2024…

大日志精选案例四:某省级大数据集团日志审计优化实战解析

“在集团日常运营中&#xff0c;数据安全始终是我们关注的重点。过去&#xff0c;数据量大、处理速度慢&#xff0c;导致日志数据难以迅速获取和分析&#xff0c;影响业务决策。但自从引入聚铭大日志解决方案后&#xff0c;系统日志和用户行为数据都得到了高效处理与存储。该方…

接口的总结与面试题

接口本身不能创建对象&#xff0c;只能创建接口的实现类对象&#xff0c;接口类型的变量可以与实现类对象构成多态引用。 声明接口用interface&#xff0c;接口的成员声明有限制&#xff1a; &#xff08;1&#xff09;公共的静态常量 &#xff08;2&#xff09;公共的抽象方…

【氮化镓】GaN SP-HEMT的栅极可靠性

概括总结&#xff1a; 本文研究了氮化镓&#xff08;GaN&#xff09;肖特基型p-栅高电子迁移率晶体管&#xff08;GaN SP-HEMT&#xff09;的栅极鲁棒性和可靠性&#xff0c;通过一种新的电路方法评估了在实际转换器中栅极电压&#xff08;VGS&#xff09;过冲波形的栅极电压应…

基于注意力整合的超声图像分割信息在乳腺肿瘤分类中的应用

基于注意力整合的超声图像分割信息在乳腺肿瘤分类中的应用 摘要引言方法 Segmentation information with attention integration for classification of breast tumor in ultrasound image 摘要 乳腺癌是世界范围内女性最常见的癌症之一。基于超声成像的计算机辅助诊断&#x…

NLP学习路线总结

自然语言处理&#xff08;Natural Language Processing&#xff0c;NLP&#xff09;是人工智能和语言学领域的一部分&#xff0c;它旨在让计算机能够理解、解释和生成人类语言。NLP学习路线可以大致分为以下几个步骤&#xff1a; 1. 基础知识准备 - 计算机科学知识&#xff1a…

贝锐蒲公英企业路由器双机热备,保障异地组网可靠、不中断

对于关键业务&#xff0c;比如&#xff1a;在线支付系统、远程医疗监控系统、重要数据中心等&#xff0c;一旦网络发生故障&#xff0c;可能导致巨大的损失或影响&#xff0c;因此需确保网络拥有极高的可靠性、稳定性和容错能力。 面对此类场景和需求&#xff0c;贝锐蒲公英异…

ubuntu18.04图形界面卡死,鼠标键盘失灵, 通过MAC共享网络给Ubuntu解决!

ubuntu18.04图形界面卡死&#xff0c;鼠标键盘失灵&#xff0c; 通过MAC共享网络给Ubuntu解决&#xff01; 1. 尝试从卡死的图形界面切换到命令行界面2. 进入bios和grub页面3. 更改Grub中的设置&#xff0c;以进入命令行4. 在命令行页面解决图形界面卡死的问题5. Mac共享WI-FI网…

中兴天机A31 A31PRO 5G zte A2122H te A2022H 解锁BootLoader root权限 教程magisk,原厂刷机包

zte A2122H P768A02 zte A2022H P875A02 中兴天机A31 A31PRO 5G zte A2122H te A2022H 解锁BootLoader root教程magisk&#xff0c;原厂刷机包 感谢 某大神支持&#xff0c;已经解锁root 刷了面具&#xff1b; 中兴天机A31 A31PRO 5G zte A2122H te A2022H 解锁BootLoad…

Vue3:Pinia简介及环境搭建

一、简介 Pinia是Vue3中的状态管理工具&#xff0c;类似与Vue2中的Vuex框架的作用 二、环境搭建 1、安装 npm install pinia2、配置 main.ts import {createApp} from vue import App from ./App.vue // 第一步&#xff1a;引入pinia import {createPinia} from piniacons…

二维动画制作软件 Animate 2024 for mac激活版

Animate 2024 for Mac是一款功能强大的二维动画制作软件&#xff0c;专为Mac用户打造。它提供了丰富的动画编辑功能&#xff0c;使用户能够轻松创建出生动逼真的动画作品。无论是短片、广告还是游戏等应用领域&#xff0c;Animate 2024都能发挥出出色的表现。 软件下载&#xf…

【运输层】网络数据报协议 UDP

目录 1、UDP 的特点 2、UDP 的首部格式 UDP 只在 IP 协议之上增加了很少的一些功能&#xff0c;比如复用、分用以及差错检测等。 1、UDP 的特点 UDP是无连接的&#xff0c;即发送数据之前不需要建立连接&#xff0c;因此减少了开销和发送数据之前的时延。 UDP使用尽最大努力…

Linux是什么,该如何学习

&#x1f407;明明跟你说过&#xff1a;个人主页 &#x1f3c5;个人专栏&#xff1a;《Linux &#xff1a;从菜鸟到飞鸟的逆袭》 &#x1f3c5; &#x1f516;行路有良友&#xff0c;便是天堂&#x1f516; 目录 一、引言 1、Linux的起源与发展 2、Linux在现代计算机领域…

Delphi编写的图片查看器

UNIT Unit17;INTERFACEUSESWinapi.Windows, Winapi.Messages, System.SysUtils, System.Variants,System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs,Vcl.StdCtrls, Vcl.ExtDlgs, Vcl.ExtCtrls, Vcl.Imaging.jpeg; //注意&#xff1a;要加入jpej 否侧浏览图…

创新指南|贝恩的产品经理RAPID框架:解决问题的分步指南,使决策过程既高效又民主

您是否曾发现自己陷入项目的阵痛之中&#xff0c;决策混乱、角色不明确、团队成员之间的冲突不断升级&#xff1f;作为产品经理&#xff0c;驾驭这艘船穿过如此汹涌的水域可能是令人畏惧的。应对这些挑战的关键在于采用清晰、结构化的决策方法。输入贝恩的 RAPID 框架&#xff…

Jackson @JsonUnwrapped注解扁平化 序列化反序列化数据

参考资料 Jackson 2.x 系列【7】注解大全篇三JsonUnwrapped 以扁平的数据结构序列化/反序列化属性Jackson扁平化处理对象 目录 一. 前期准备1.1 前端1.2 实体类1.3 Controller层 二. 扁平化序列反序列化数据2.1 序列化数据2.2 反序列化数据 三. 前缀后缀处理属性同名四. Map数…