C++链表双杰:list与forward_list

在C++容器的世界里,当我们需要频繁地在序列中间进行插入和删除时,基于数组的 vector 会显得力不从心。这时,链表结构就闪亮登场了。STL提供了两种链表容器:功能全面的双向链表 std::list 和极致轻量化的单向链表 std::forward_list

你是否好奇它们为何在某些场景下性能远超 vector?又为何在另一些场景下又应避免使用?list 和 forward_list 之间又该如何抉择?今天,我们将深入链表的微观世界,通过清晰的解释、生动的比喻和实用的代码,为你彻底揭开它们的神秘面纱。


第一部分:核心概念与底层实现

什么是链表?

链表是一种物理存储单元上非连续、非顺序的存储结构。数据元素的逻辑顺序是通过链表中的指针链接次序实现的。

与 vector 的直观对比:

  • vector(动态数组):像一列火车。车厢(元素)是连续连接的。找到车头就知道所有车厢的位置,快速找到第n节车厢(随机访问)。但想在中部插入或移除一节车厢,需要移动后面所有车厢,非常耗时。

  • list / forward_list(链表):像寻宝游戏。每个藏宝点(节点)只知道下一个藏宝点在哪里(指针)。从一个点开始,你必须按线索逐个寻找(顺序访问)。但你想在中间添加或移除一个藏宝点,只需修改它前后点的线索即可,无需移动其他所有点。

底层实现:节点(Node)

链表的每个元素都存储在一个独立的节点中。每个节点至少包含两个部分:

  1. 数据(data):存储的实际值。

  2. 指针(pointer):指向下一个(和上一个)节点的地址。

// std::list<double> 的节点可能类似这样:
struct ListNode {double data;        // 存储的数据ListNode* next;     // 指向下一个节点ListNode* prev;     // 指向上一个节点
};// std::forward_list<int> 的节点可能类似这样:
struct ForwardListNode {int data;           // 存储的数据ForwardListNode* next; // 仅指向下一个节点
};

第二部分:std::list - 功能全面的双向链表

定义与特性

std::list 是一个双向链表容器。它允许在常量时间内在序列的任何位置进行插入和删除操作。

  • 核心特性

    • 双向遍历:每个节点都有指向前驱和后继的指针,支持从前往后和从后往前的遍历。

    • 高效的中间操作:在已知位置插入或删除元素的时间复杂度是 O(1)

    • 非连续存储:元素散落在内存中,因此不支持随机访问(如 list[5] 是错误的)。

    • 迭代器稳定性:插入操作不会使任何现有迭代器失效;删除操作仅使被删除元素的迭代器失效。这是它与 vector 最大的不同之一。

C++ 代码示例
#include <iostream>
#include <list>
#include <algorithm> // for std::findint main() {std::list<int> myList = {5, 2, 8, 1, 3}; // 初始化// 1. 在头部和尾部插入元素 (O(1))myList.push_front(10);myList.push_back(20);// 2. 在中间插入元素 (O(1))auto it = std::find(myList.begin(), myList.end(), 8); // 找到值为8的位置if (it != myList.end()) {myList.insert(it, 99); // 在8之前插入99}// 3. 遍历链表 (只能使用迭代器,无随机访问)std::cout << "List contents: ";for (const int& value : myList) { // 范围for循环std::cout << value << " ";}std::cout << std::endl;// 4. 删除元素 (O(1))myList.remove(2); // 删除所有值为2的元素myList.pop_front(); // 删除头部元素myList.pop_back();  // 删除尾部元素// 5. 强大的链表操作:拼接(splice) - 将另一个链表的一部分移动到本链表std::list<int> otherList = {100, 200, 300};auto pos = myList.begin();std::advance(pos, 2); // 将迭代器pos前进2步myList.splice(pos, otherList); // 将otherList的所有元素移动到pos之前// 此时,otherList变为空std::cout << "Size of otherList after splice: " << otherList.size() << std::endl;return 0;
}
应用场景
  • 频繁在序列任意位置进行插入/删除:如实现一个消息队列,需要频繁地从中部移除或添加任务。

  • 迭代器稳定性要求高:你需要在进行大量插入删除操作后,之前获取的迭代器(除了指向被删除元素的)仍然有效。

  • 需要实现复杂的数据结构:如LRU缓存淘汰算法(使用list和unordered_map组合)。


第三部分:std::forward_list - 极致轻量的单向链表

定义与特性

std::forward_list 是C++11引入的单向链表容器。它设计的目标是极致的内存效率。

  • 核心特性

    • 单向遍历:每个节点只有一个指向下一个节点的指针,因此只能从前往后遍历

    • 更小的开销:每个节点比 list 的节点少一个指针,内存占用更小。

    • API设计特殊:为了极致优化,它甚至不提供 size() 方法,因为维护一个计数器会有开销。获取大小需要 O(n) 时间遍历计数。它的插入和删除操作通常需要指向前一个元素的迭代器

C++ 代码示例
#include <iostream>
#include <forward_list>int main() {std::forward_list<int> flist = {1, 2, 3};// 1. 在头部插入元素 (O(1)) - 这是最自然的操作flist.push_front(0);// 2. 在指定位置之后插入 (O(1)) - 这是主要操作方式auto it = flist.begin(); // it指向0flist.insert_after(it, 99); // 在0之后插入99 -> [0, 99, 1, 2, 3]// 3. 遍历 (和list一样)std::cout << "Forward_list contents: ";for (const int& val : flist) {std::cout << val << " ";}std::cout << std::endl;// 4. 删除指定位置之后的元素 (O(1))it = flist.begin(); // it指向0flist.erase_after(it); // 删除0后面的元素(99) -> [0, 1, 2, 3]// 5. 获取大小?需要遍历!int count = 0;for (auto it = flist.begin(); it != flist.end(); ++it) { ++count; }std::cout << "Size is approximately: " << count << std::endl; // 不推荐频繁使用return 0;
}
应用场景
  • 对内存空间要求极度苛刻的嵌入式系统或底层程序。

  • 只需要单向遍历,且插入删除操作多发生在已知节点的后面

  • 实现哈希桶(Hash Bucket)或邻接表(图的表示),这些结构本身就不需要反向遍历。


第四部分:终极对决:对比与选择

为了让你一目了然,我们通过表格来进行全面对比。

list vs forward_list vs vector 特性对比表
特性std::list (双向链表)std::forward_list (单向链表)std::vector (动态数组)
底层数据结构双向链表单向链表动态数组
内存布局非连续非连续连续
随机访问不支持,O(n)不支持,O(n)支持,O(1)
头部插入/删除O(1)O(1)O(n)
尾部插入/删除O(1)O(n)¹O(1) 均摊
中间插入/删除O(1) (已知位置)O(1) (已知前驱位置)O(n)
迭代器类型双向迭代器前向迭代器随机访问迭代器
迭代器稳定性 (插入不失效,删除仅当前失效) (扩容全部失效)
内存开销大 (每个元素2指针)小 (每个元素1指针)小 (近乎0额外开销)
缓存友好性极好
特殊成员size()push_backpop_backsplice无 size()insert_aftererase_afterreserve()capacity()data()

forward_list 需要O(n)时间找到尾部,因此尾部操作不是它的设计目的。

如何选择?决策指南

结论

没有完美的容器,只有最适合的场景。

  • std::vector 是通用之王,在大多数情况下都是默认的最佳选择。

  • std::list 是中间操作大师,当你的业务核心是频繁的、不可预测的插入和删除时,它就是你的利器。

  • std::forward_list 是空间优化专家,在资源极其受限且需求匹配的特殊场景下,它无可替代。

理解它们的内在原理和性能特征,就像为你的代码工具箱选择了最合适的那把螺丝刀,让你能够写出既高效又优雅的程序。下次面临选择时,不妨先问问自己:“我最频繁的操作是什么?” 答案自然会浮现。

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

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

相关文章

Ruoyi-vue-plus-5.x第一篇Sa-Token权限认证体系深度解析:1.4 Sa-Token高级特性实现

&#x1f44b; 大家好&#xff0c;我是 阿问学长&#xff01;专注于分享优质开源项目解析、毕业设计项目指导支持、幼小初高的教辅资料推荐等&#xff0c;欢迎关注交流&#xff01;&#x1f680; Sa-Token高级特性实现 前言 在前面的文章中&#xff0c;我们学习了Sa-Token的…

Linux 服务器初始化解析和ssh密钥交换的介绍

目录 2. SSH 基于密钥交换的介绍和原理 2.1 核心优势 2.2 密钥交换原理&#xff08;非对称加密体系&#xff09; 2.3 基础配置步骤 3. 服务器初始化 3.1 安装 yum 网络源 3.1.1 背景说明 3.1.2 实操步骤 3.2 安装运维的必备工具 3.2.1 工具清单 3.2.2 批量安装命令 …

web渗透ASP.NET(Webform)反序列化漏洞

web渗透ASP.NET(Webform)反序列化漏洞1&#xff09;ASP.NET(Webform)反序列化漏洞ASP.NET(Webform) 反序列化漏洞的核心触发点是 Webform 框架中的VIEWSTATE参数 —— 该参数用于存储页面控件状态数据&#xff0c;默认以 Base64 编码传输&#xff0c;内部包含序列化的对象数据。…

Android FrameWork - 开机启动 SystemServer 进程

基于安卓 12 源码分析相关类&#xff1a;frameworks/base/core/java/com/android/internal/os/ZygoteInit.java frameworks/base/core/java/com/android/internal/os/Zygote.java frameworks/base/core/java/com/android/internal/os/RuntimeInit.java frameworks/base/service…

C++:list容器--模拟实现(下篇)

1. 模拟实现 list 一些常用接口// list.h #pragma once #include <assert.h> #include "Iterator.h"namespace room {template<class T>struct list_node{list_node<T>* _next;list_node<T>* _prev;T _data;list_node(const T& x T()):…

边缘计算:一场由物理定律发起的“计算革命”

专栏引言:在前面的文章中,我们探讨了云计算如何将计算资源变成了“数字水电煤”,构建了一个强大的中心化数字帝国。然而,当这个帝国试图将它的触角伸向物理世界的每一个角落时,却遭遇了两位“上古之神”的无情阻击——光速与带宽。今天,我们将聚焦于一场由物理定律发起的…

量化模型部署工具llama.cpp

量化模型部署工具llama.cppllama.cppllama.cpp 是什么使用场景是什么如何使用&#xff1f;第 1 步&#xff1a;获取量化模型第 2 步&#xff1a;编译 llama.cpp第 3 步&#xff1a;运行推理完整 Demo&#xff1a;与 Llama 3 对话进阶使用&#xff1a;Python 集成总结概念解释1.…

【光照】[光照模型]发展里程碑时间线

【从UnityURP开始探索游戏渲染】专栏-直达 图形学光照模型发展史&#xff1a;技术演进与里程碑 section 基础奠基期(1960s-1970s) 1967 &#xff1a; Lambert模型(漫反射) - Bui Tuong Phong提出1971 &#xff1a; Gouraud着色 - Henri Gouraud发明顶点插值着色1973 &#xf…

【从零开始java学习|第十篇】面向对象

目录 一、面向对象介绍 二、类和对象 1. 类&#xff08;Class&#xff09;&#xff1a;对象的模板 2. 对象&#xff08;Object&#xff09;&#xff1a;类的实例 三、封装 1. 封装的概念 2. 封装的优势 四、就近原则和 this 关键字 1. 就近原则 2. this 关键字 五、…

Spark算子调优

Spark中可用下面的算子对数据计算进行优化处理&#xff0c;包括&#xff1a; mapPartition&#xff1a;一次处理一个分区数据&#xff0c;能够使用mapPartition的尽量使用&#xff0c;但是使用时会一次性读取整个分区数据到内存&#xff0c;占内存很大&#xff0c;同理还有fore…

码农特供版《消费者权益保护法》逆向工程指北——附源码级注释与异常处理方案

尊敬的审核&#xff1a; 本人文章《码农特供版〈消费者权益保护法〉逆向工程指北——附源码级注释与异常处理方案》 1. 纯属技术交流&#xff0c;无任何违法内容 2. 所有法律引用均来自公开条文 3. 请依据《网络安全法》第12条“不得无故删除合法内容”处理 附&#xff1a;本文…

MQTT 连接建立与断开流程详解(二)

三、核心机制与最佳实践&#xff08;一&#xff09;会话管理与 QoS 保障Clean Session vs 持久会话&#xff1a;在 MQTT 连接中&#xff0c;会话管理是一个重要的概念&#xff0c;其中 Clean Session 和持久会话是两种不同的会话模式。Clean Session&#xff0c;当设置为 1 时&…

[光学原理与应用-332]:ZEMAX - 序列模式与非序列模式的本质、比较

序列模式&#xff08;Sequential Mode&#xff09;与非序列模式&#xff08;Non-Sequential Mode&#xff09;是ZEMAX光学设计软件中的两种核心设计模式&#xff0c;二者在光路定义、分析工具、应用场景等方面存在本质差异。以下是两者的详细比较&#xff1a;一、本质差异光路定…

WeakAuras Lua Script (My Version)

分享下我的WA的简约配置&#xff0c;大多数都是团队框架高亮&#xff0c;辅助大脚DBM监控 表格&#xff1a; WeakAuras Lua Script &#xff1c;BiaoGe&#xff1e;_wa拍卖字符串-CSDN博客 ICC 监控&#xff0c;只要团队框架监控 WeakAuras Lua Script ICC &#xff08;Barne…

【Python+requests】解决Python requests中的ProxyError:SSL版本错误问题详解

解决Python requests中的ProxyError&#xff1a;SSL版本错误问题详解 在使用Python进行网络请求时&#xff0c;很多人都会用到requests库配合代理服务器进行调试或抓包。但有时会遇到令人困惑的ProxyError&#xff0c;尤其是伴随SSLError: [SSL: WRONG_VERSION_NUMBER]这样的错…

基于deepseek的Spring boot入门

一次跟着deepseek记笔记的尝试&#xff0c;由于CSDN没有思维导图&#xff0c;只能按层级记录提问 如果我想知道一个springboot项目的基本结构&#xff0c;比如用到了哪些组件&#xff0c;入口在哪&#xff0c;数据库配置是怎样的 应该从哪里开始 springboot有哪些常用注解 一个…

macOS 15.6 ARM golang debug 问题

前言 最近使用macmini m4在使用golang debug发现一些奇怪的问题&#xff0c;debug到c代码&#xff0c;莫名其妙&#xff0c;而且不知道什么原因&#xff0c;知道搜索查询&#xff0c;才发现是苹果的Command Line Tools 的锅&#xff0c;macOS 15果然是一堆bug&#xff0c;毕竟…

有个需求:切换车队身份实现Fragment的Tab隐藏显示(车队不显示奖赏)

核心实现&#xff1a; 1使用mmkv保存切换的身份 2借助eventbus实现通知Fragment的tab更新private void switchFleet(boolean isMore, EnterpriseInfo enterpriseInfo) {if (isMore) {tvSwitchFleetTitle.setText(getText(R.string.switch_to_other_accounts));} else {tvSwitch…

在 Android Studio 中修改 APK 启动图标(2025826)

在 Android Studio 中修改 Android 12 应用图标可以按照以下步骤进行&#xff1a;1、准备图标资源准备一个启动图标&#xff08;建议使用 SVG 格式或高分辨率 PNG&#xff0c;推荐尺寸为 512x512 像素&#xff09;图标应符合 Android 12 的设计规范&#xff08;自适应图标&…

Linux三剑客grep-sed-awk

linux三剑客-grep、sed、awk 文章目录linux三剑客-grep、sed、awk1.正则表达式1.1正则表达式&#xff1f;1.2应用场景&#xff1f;-谁可以用&#xff1f;1.3正则注意事项&#xff08;避免90%以上的坑&#xff09;1.4正则符号1.5正则VS通配符2.基础正则2.1 ^ 以...开头的行2.2 $…