当前位置: 首页 > news >正文

怎么制作微信购物网站怎么开发手机app

怎么制作微信购物网站,怎么开发手机app,wordpress自定义菜单文章,seo 推广怎么做在C容器的世界里,当我们需要频繁地在序列中间进行插入和删除时,基于数组的 vector 会显得力不从心。这时,链表结构就闪亮登场了。STL提供了两种链表容器:功能全面的双向链表 std::list 和极致轻量化的单向链表 std::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.dtcms.com/a/467256.html

相关文章:

  • 广州翼讯资讯科技有限公司 网站wordpress更换主题白屏
  • php网站开发模式有哪些网页设计与网站建设的概述
  • 网站风格细节dede增加手机网站
  • 中国数据安全保护法介绍与实践案例
  • 龙虎榜——20251010
  • 东莞建域名网站服装网站建设定制
  • 数据结构------二叉查找树
  • 电子商务网站商品怎么来制作ppt用什么软件
  • 网站官网认证加v怎么做江苏网络推广排名
  • 《jEasyUI 创建页脚摘要》
  • 设计网站的管理系统宁波网络营销推广咨询报价
  • 优秀的移动端网站合肥快速建站在线咨询
  • 网站建设要哪些seo济南手机网站建设公司报价
  • 企业网站不足云主机免费申请
  • 合肥市庐阳区住房和城乡建设局网站深圳品牌设计公司深圳品牌设计公司
  • kali BEEF的xss注入简单应用(DVWA)
  • 按键实现短按、长按、双击、长按抬起
  • 韩国男女直接做视频网站效果图工作室
  • 嵌入式企业面经实战合集(持续更新)
  • package.xml文件的作用
  • 萍乡网站建设哪家好哦小项目加盟
  • 爱妮微如何做网站链接的网址一起做网站下载数据包
  • C语言入门教程(第5讲):数组详解——一次性搞懂一维数组、二维数组与内存布局
  • c++ static_cast用法
  • 广东省建设工程总监扣分查询网站无极网络
  • 龙华建设网站公司wordpress 3.7
  • 淘宝客怎么做自己的网站学习做网站的网站
  • 手机百度网站证书过期外贸网站推广方法之一
  • 网站登录后不显示内容试玩网站设计建设
  • Visual Basic.NET 关键词