STL-list容器的使用
在C++标准库中,std::list
是一个双向链表容器,提供高效的插入和删除操作,尤其适用于需要频繁在容器中间进行插入和删除元素的场景。与其他序列容器(如 std::vector
和 std::deque
)相比,std::list
有其独特的优势和使用场景。
目录
- 简介
- 主要特点
- 基本用法
- 声明和初始化
- 常用操作
- 迭代器
- 性能比较
- 高级功能
- splice
- merge
- sort
- reverse
- 使用场景
- 示例代码
- 注意事项
- 总结
简介
std::list
是C++标准模板库(STL)中的一个容器,基于双向链表实现。它允许在任意位置高效地插入和删除元素,但不支持随机访问(如通过索引访问元素)。适用于需要频繁进行中间插入和删除操作,而不需要快速随机访问的场景。
主要特点
- 双向链表:每个元素包含指向前后元素的指针,支持双向遍历。
- 动态大小:容器大小可根据需要动态调整,无需预先分配。
- 高效的插入和删除:在任何位置插入或删除元素的时间复杂度为常数时间
O(1)
,前提是已知位置。 - 不支持随机访问:无法像
std::vector
那样通过索引直接访问元素。 - 稳定的迭代器:插入和删除操作不会使迭代器失效,除非删除了迭代器所指向的元素。
基本用法
声明和初始化
#include <iostream>
#include <list>
#include <string>int main() {// 声明一个空的 liststd::list<int> intList;// 使用初始化列表初始化std::list<std::string> stringList = {"Apple", "Banana", "Cherry"};// 使用拷贝构造函数std::list<std::string> copiedList(stringList);// 使用指定数量和初始值初始化std::list<int> filledList(5, 100); // 包含5个100return 0;
}
常用操作
插入元素
- push_back 和 push_front:在末尾和开头插入元素。
- insert:在指定位置插入元素。
#include <list>
#include <iostream>int main() {std::list<int> myList = {1, 2, 3};// 在末尾插入myList.push_back(4); // {1, 2, 3, 4}// 在开头插入myList.push_front(0); // {0, 1, 2, 3, 4}// 在指定位置插入auto it = myList.begin();++it; // 指向1myList.insert(it, 100); // {0, 100, 1, 2, 3, 4}// 输出列表内容for (const auto& num : myList) {std::cout << num << " ";}// 输出: 0 100 1 2 3 4return 0;
}
删除元素
- pop_back 和 pop_front:删除末尾和开头元素。
- erase:删除指定位置的元素。
- remove:删除所有匹配值的元素。
#include <list>
#include <iostream>int main() {std::list<int> myList = {0, 100, 1, 2, 3, 4};// 删除末尾元素myList.pop_back(); // {0, 100, 1, 2, 3}// 删除开头元素myList.pop_front(); // {100, 1, 2, 3}// 删除特定元素(通过迭代器)auto it = myList.begin();++it; // 指向1myList.erase(it); // {100, 2, 3}// 删除所有值为100的元素myList.remove(100); // {2, 3}// 输出列表内容for (const auto& num : myList) {std::cout << num << " ";}// 输出: 2 3return 0;
}
访问元素
由于 std::list
不支持随机访问,因此不能使用下标操作符 []
。可以使用迭代器或范围 for
循环访问元素。
#include <list>
#include <iostream>int main() {std::list<std::string> fruits = {"Apple", "Banana", "Cherry"};// 使用迭代器访问for (std::list<std::string>::iterator it = fruits.begin(); it != fruits.end(); ++it) {std::cout << *it << " ";}// 输出: Apple Banana Cherry// 使用范围 for 循环for (const auto& fruit : fruits) {std::cout << fruit << " ";}// 输出: Apple Banana Cherryreturn 0;
}
修改元素
#include <list>
#include <iostream>int main() {std::list<int> numbers = {1, 2, 3, 4, 5};// 修改所有元素for (auto& num : numbers) {num *= 10; // {10, 20, 30, 40, 50}}// 输出修改后的列表for (const auto& num : numbers) {std::cout << num << " ";}// 输出: 10 20 30 40 50return 0;
}
迭代器
std::list
提供双向迭代器(std::list::iterator
和 std::list::const_iterator
),允许从前向后或从后向前遍历元素。
#include <list>
#include <iostream>int main() {std::list<int> myList = {1, 2, 3, 4, 5};// 前向迭代std::cout << "前向迭代: ";for (std::list<int>::iterator it = myList.begin(); it != myList.end(); ++it) {std::cout << *it << " ";}// 输出: 1 2 3 4 5// 反向迭代std::cout << "\n反向迭代: ";for (std::list<int>::reverse_iterator rit = myList.rbegin(); rit != myList.rend(); ++rit) {std::cout << *rit << " ";}// 输出: 5 4 3 2 1return 0;
}
性能比较
std::list
vs std::vector
特性 | std::list | std::vector |
---|---|---|
随机访问 | 不支持 | 支持 O(1) 时间复杂度 |
插入/删除 | 中间插入/删除 O(1) (已知位置) | 中间插入/删除 O(n) |
插入/删除(末尾) | O(1) | 平均 O(1) ,摊销 |
内存使用 | 每个元素需要额外的指针 | 更紧凑,元素连续存储 |
缓存局部性 | 差 | 良好,连续存储提高缓存命中率 |
遍历 | 线性时间 | 线性时间,缓存友好 |
空间效率 | 较低 | 较高 |
选择建议
-
使用
std::list
:- 需要频繁在容器中间插入和删除元素。
- 不需要随机访问元素。
- 需要稳定的迭代器,不希望插入/删除操作导致迭代器失效。
-
使用
std::vector
:- 需要快速随机访问元素。
- 插入和删除操作主要在末尾进行。
- 更好的缓存性能,提高遍历效率。
高级功能
splice
splice
方法用于在两个 std::list
之间移动元素,无需复制或移动元素本身,效率极高。
#include <list>
#include <iostream>int main() {std::list<int> list1 = {1, 2, 3};std::list<int> list2 = {4, 5, 6};// 将 list2 的所有元素移动到 list1 的末尾list1.splice(list1.end(), list2);// list1: {1, 2, 3, 4, 5, 6}// list2: {}// 输出 list1std::cout << "list1: ";for (const auto& num : list1) {std::cout << num << " ";}std::cout << "\nlist2: ";for (const auto& num : list2) {std::cout << num << " ";}return 0;
}
splice 的重载形式
-
全列表移动:
void splice(iterator pos, list& other);
-
移动单个元素:
void splice(iterator pos, list& other, iterator it);
-
移动范围的元素:
void splice(iterator pos, list& other, iterator first, iterator last);
merge
merge
方法用于合并两个已排序的 std::list
,结果仍然有序。需要确保两个列表在合并前已经排序。
#include <list>
#include <iostream>int main() {std::list<int> list1 = {1, 3, 5};std::list<int> list2 = {2, 4, 6};// 合并并排序list1.merge(list2);// list1: {1, 2, 3, 4, 5, 6}// list2: {}// 输出 list1std::cout << "Merged list1: ";for (const auto& num : list1) {std::cout << num << " ";}return 0;
}
sort
sort
方法对 std::list
中的元素进行排序。由于 std::list
是链表结构,不支持快速随机访问,内部使用归并排序算法,时间复杂度为 O(n log n)
。
#include <list>
#include <iostream>int main() {std::list<int> numbers = {4, 2, 5, 1, 3};// 排序numbers.sort();// 输出排序后的列表for (const auto& num : numbers) {std::cout << num << " ";}// 输出: 1 2 3 4 5return 0;
}
reverse
reverse
方法用于反转 std::list
中的元素顺序。
#include <list>
#include <iostream>int main() {std::list<int> numbers = {1, 2, 3, 4, 5};// 反转numbers.reverse();// 输出反转后的列表for (const auto& num : numbers) {std::cout << num << " ";}// 输出: 5 4 3 2 1return 0;
}
使用场景
std::list
适用于以下场景:
- 频繁在中间插入和删除元素:如实现队列、双端队列(deque)等数据结构。
- 需要稳定的迭代器:插入和删除操作不会使其他迭代器失效。
- 不需要随机访问:主要进行顺序访问和操作。
示例:实现一个任务队列,频繁添加和移除任务。
#include <list>
#include <iostream>
#include <string>int main() {std::list<std::string> taskQueue;// 添加任务taskQueue.push_back("Task 1");taskQueue.push_back("Task 2");taskQueue.push_back("Task 3");// 插入一个高优先级任务taskQueue.push_front("High Priority Task");// 处理任务while (!taskQueue.empty()) {std::cout << "Processing: " << taskQueue.front() << std::endl;taskQueue.pop_front();}return 0;
}
示例代码
下面是一个综合示例,展示 std::list
的各种操作。
#include <iostream>
#include <list>
#include <string>int main() {// 声明和初始化std::list<std::string> fruits = {"Apple", "Banana", "Cherry"};// 插入元素fruits.push_back("Date");fruits.push_front("Apricot");auto it = fruits.begin();std::advance(it, 2); // 指向第三个元素fruits.insert(it, "Blueberry"); // 在第三个元素前插入// 输出当前列表std::cout << "Fruits list: ";for (const auto& fruit : fruits) {std::cout << fruit << " ";}std::cout << std::endl;// 输出: Fruits list: Apricot Apple Blueberry Banana Cherry Date // 删除元素fruits.remove("Banana"); // 删除所有"Banana"// 使用迭代器删除it = fruits.begin();++it; // 指向第二个元素fruits.erase(it); // 删除第二个元素// 输出修改后的列表std::cout << "After deletion: ";for (const auto& fruit : fruits) {std::cout << fruit << " ";}std::cout << std::endl;// 输出: After deletion: Apricot Blueberry Cherry Date // 排序fruits.sort();std::cout << "After sorting: ";for (const auto& fruit : fruits) {std::cout << fruit << " ";}std::cout << std::endl;// 输出: After sorting: Apricot Blueberry Cherry Date // 反转fruits.reverse();std::cout << "After reversing: ";for (const auto& fruit : fruits) {std::cout << fruit << " ";}std::cout << std::endl;// 输出: After reversing: Date Cherry Blueberry Apricot // 清空列表fruits.clear();std::cout << "After clearing, size: " << fruits.size() << std::endl;// 输出: After clearing, size: 0return 0;
}
注意事项
- 不支持随机访问:
std::list
不支持通过索引访问元素。需要通过迭代器或范围for
循环进行遍历。 - 内存开销:每个元素存储额外的指针(前向和后向),相比
std::vector
内存开销更大。 - 缓存局部性差:由于元素不连续存储,遍历性能不如
std::vector
,特别是在需要频繁遍历的场景下。 - 适用场景有限:在大多数需要顺序容器的场景下,
std::vector
更高效,只有在特定情况下,如频繁的中间插入和删除,才需要考虑使用std::list
。 - 操作复杂性:某些操作,如排序、合并,虽然
std::list
支持,但相比std::vector
的内置算法,可能需要更多的注意。
总结
std::list
是一个基于双向链表的容器,适用于需要频繁在中间位置进行插入和删除操作的场景。它提供高效的插入和删除性能,但不支持随机访问,内存开销较大,且缓存局部性差。在选择使用 std::list
之前,应权衡其优势和劣势,并考虑是否有其他更适合的容器(如 std::vector
或 std::deque
)。
使用建议:
- 优先考虑使用
std::vector
,因为它在大多数情况下性能更优。 - 只有在需要频繁进行中间插入和删除,且不需要随机访问时,才考虑使用
std::list
。 - 考虑使用其他更现代的容器或数据结构,如
std::forward_list
(单向链表),或使用std::deque
进行高效的两端操作。
希望这个详细的介绍能帮助你更好地理解和使用C++中的 std::list
容器!
相关文章:
STL-list容器的使用
在C标准库中,std::list 是一个双向链表容器,提供高效的插入和删除操作,尤其适用于需要频繁在容器中间进行插入和删除元素的场景。与其他序列容器(如 std::vector 和 std::deque)相比,std::list 有其独特的优…...
java中线程与集合的面试题
在 Java 面试中,线程和集合相关的知识是非常常见的考察点。以下是几个典型的问题及答案: 线程相关面试题 什么是线程? 线程是操作系统能够进行运算调度的最小单位。它被包含在进程之中,是进程中的实际运作单位。一个进程可以有多…...
第十五章 IRIS 进程之间的通信
文章目录 第十五章 IRIS 进程之间的通信介绍指定作业间通信设备的内存缓冲区禁用作业间通信缓冲区 作业间通信设备编号设备编号 IJC 设备的 I/O 命令OPEN命令device 设备timeout 暂停 第十五章 IRIS 进程之间的通信 本页介绍如何在两个或多个 IRIS 数据平台进程之间建立通信。…...

设计者模式之策略模式
前言 在软件构建过程中,某些对象使用的算法可能多种多样,经常改变,如果将这些算法都写在对象中,将会使对象变得异常复杂;而且有时候支持不频繁使用的算法也是一个性能负担。 如何在运行时根据需要透明地更改对象的算…...

STM32H750 COMP模拟比较器
STM32H750 COMP模拟比较器 🔖STM32H750内置两个超低功耗比较器通道(COMP1 和 COMP2). 📄功能应用: 在模拟信号的触发下从低功耗模式唤醒模拟信号调理与定时器的 PWM 输出结合使用时,构成逐周期电流控制环路…...
openresty入门教程:rewrite_by_lua_block
在OpenResty中,rewrite_by_lua_block 是一个强大的工具,它允许你在Nginx的rewrite阶段执行Lua脚本。这个阶段在Nginx处理请求的早期发生,通常用于修改请求URI、请求参数、请求头等,或者根据某些条件执行重定向、返回特定响应等。 …...

Java 并发编程学习笔记
参考资料: JAVA并发专题 - 终有救赎的专栏 - 掘金 Java并发编程学习路线(建议收藏��) | Java程序员进阶之路x沉默王二 面试题目: JUC第一讲:Java并发知识体系详解 面试题汇总(P6熟练 P7精通…...

【SpringMVC】——Cookie和Session机制
阿华代码,不是逆风,就是我疯 你们的点赞收藏是我前进最大的动力!! 希望本文内容能够帮助到你!! 目录 一:实践 1:获取URL中的参数 (1)PathVariable 2&…...

[产品管理-60]:产品的情感化设计与常用工具:感性工学、情感分析、神经网络法、微软反应卡、突发情绪法
目录 一、概述 1、情感化设计的三个层次 2、情感化设计在产品中的应用 3、情感化设计的案例 4、情感化设计的意义 二、常见工具 1、感性工学 (情商) 2、情感分析 3、神经网络法 4、微软反应卡 5、突发情绪法 一、概述 产品的情感化设计是一种…...
uniapp 小程序 周选择器
这里贴出来的是子组件的代码,父组件只是打开了一下popup // 打开了一下popup $refs.popup.open(bottom)如果不想用子组件的话,直接打开popup就可以用<template><uni-popup ref"popup" type"bottom" background-color&quo…...
Android笔记(三十二):封装一个毫秒级别倒计时View
效果 倒计时View视频 背景 业务场景需要显示带有毫秒级别的倒计时,于是自己封装一个通用的倒计时组件 源码分析 核心倒计时逻辑,主要是每隔100毫秒计算一次从开始倒计时到现在的剩余时间,并通过process接口返回出去Handler每次设置100毫秒…...

[产品管理-60]:马斯洛需求层次与产品的情感化设计
目录 一、概述 1、马斯洛需求层次理论概述 2、产品情感化设计与马斯洛需求层次的关系 3、产品情感化设计的实践案例 二、马斯洛需求层次与用户情感程度(本能、行为、反思)的关系 1、马斯洛需求层次与用户情感程度概述 2、马斯洛需求层次与用户情感…...

Python接口自动化测试自学指南(项目实战)
🍅 点击文末小卡片 ,免费获取软件测试全套资料,资料在手,涨薪更快 接口自动化测试是指通过编写程序来模拟用户的行为,对接口进行自动化测试。Python是一种流行的编程语言,它在接口自动化测试中得到了广…...
ESLint 使用教程(三):12个ESLint 配置项功能与使用方式详解
前言 在现代前端开发中,代码质量与一致性是至关重要的,ESLint 正是为此而生的一款强大工具,本文将带您详细了解 ESLint 的配置文件,并通过通俗易懂的方式讲解其主要配置项及其配置方法。此外,我们还将探讨一些高级配置…...
如何将 EDB 文件导入 Ansys HFSS 和 Ansys Q3D
EDB 文件包含有关印刷电路板 (PCB) 的基本数据,包括其布局、组件、连接性和电磁属性。将 EDB 文件导入 Ansys 工具是利用仿真和分析功能设计高效、可靠和高性能电子系统的关键步骤。在这里,我将向您展示如何将 EDB 文件导入 Ansys…...

HbuildderX运行到手机或模拟器的Android App基座识别不到设备 mac
寻找模拟器 背景: 运行的是h5,模拟器是网易MuMu。 首先检查一下是否配置dab环境,adb version 配置一下hbuilderX的adb: 将命令输出的路径配置到hbuilderx里面去,然后重启下HbuilderX。 开始安装基座…一直安装不…...

智慧流控 力行天地 | 同元软控受邀参加第十三届全国流体传动与控制学术会议
2024年10月27日-30日,由中国机械工程学会流体传动与控制分会主办的第十三届全国流体传动与控制学术会议在秦皇岛召开。大会以“智慧流控 力行天地”为主题,来自全国高校、科研院所及企事业单位的专家学者出席本次会议。 大会围绕工程应用、新型流控元件、…...
Python日志分析与故障定位
Python日志分析与故障定位 目录 📊 分布式系统日志分析:ELK Stack与Fluentd⚡ 实时日志流处理与异常检测🐍 使用Python分析并处理海量日志数据🚨 自动化故障检测与报警系统🔍 故障根因分析(Root Cause An…...

w029基于springboot的网上购物商城系统研发
🙊作者简介:拥有多年开发工作经验,分享技术代码帮助学生学习,独立完成自己的项目或者毕业设计。 代码可以查看文章末尾⬇️联系方式获取,记得注明来意哦~🌹赠送计算机毕业设计600个选题excel文件࿰…...
Uniapp全局文件执行顺序详解
Uniapp全局文件执行顺序详解 在Uni-App项目中,全局文件的执行顺序对于深入理解应用的启动和初始化流程至关重要。本文将详细阐述这些文件的执行顺序,并提供相应的示例代码,以便开发者更好地理解和应用。 1. index.html 文件描述࿱…...

Docker 离线安装指南
参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...

idea大量爆红问题解决
问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...

手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...

地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...
golang循环变量捕获问题
在 Go 语言中,当在循环中启动协程(goroutine)时,如果在协程闭包中直接引用循环变量,可能会遇到一个常见的陷阱 - 循环变量捕获问题。让我详细解释一下: 问题背景 看这个代码片段: fo…...

盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
k8s从入门到放弃之Ingress七层负载
k8s从入门到放弃之Ingress七层负载 在Kubernetes(简称K8s)中,Ingress是一个API对象,它允许你定义如何从集群外部访问集群内部的服务。Ingress可以提供负载均衡、SSL终结和基于名称的虚拟主机等功能。通过Ingress,你可…...
在HarmonyOS ArkTS ArkUI-X 5.0及以上版本中,手势开发全攻略:
在 HarmonyOS 应用开发中,手势交互是连接用户与设备的核心纽带。ArkTS 框架提供了丰富的手势处理能力,既支持点击、长按、拖拽等基础单一手势的精细控制,也能通过多种绑定策略解决父子组件的手势竞争问题。本文将结合官方开发文档,…...

【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...

大型活动交通拥堵治理的视觉算法应用
大型活动下智慧交通的视觉分析应用 一、背景与挑战 大型活动(如演唱会、马拉松赛事、高考中考等)期间,城市交通面临瞬时人流车流激增、传统摄像头模糊、交通拥堵识别滞后等问题。以演唱会为例,暖城商圈曾因观众集中离场导致周边…...