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)1O(1)?均攤
中間插入/刪除O(1)?(已知位置)O(1)?(已知前驅位置)O(n)
迭代器類型雙向迭代器前向迭代器隨機訪問迭代器
迭代器穩定性?(插入不失效,刪除僅當前失效)?(擴容全部失效)
內存開銷大 (每個元素2指針)小 (每個元素1指針)小 (近乎0額外開銷)
緩存友好性極好
特殊成員size(),?push_back,?pop_back,?splice無?size(),?insert_after,?erase_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
英文地址,請注明出處:http://en.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 $…