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

C++ STL 容器:List 深度解析与实践指南

一、List 容器概述

1.1底层结构与特性

  • 数据结构:双向循环链表(带哨兵位头结点),每个节点包含前驱指针、后继指针和数据域。
  • 核心优势
    • 高效插入 / 删除:任意位置操作时间复杂度为 O (1),无需移动元素。
    • 灵活迭代器:支持双向遍历(正向迭代器 begin/end、反向迭代器 rbegin/rend)。
  • 局限性
    • 不支持随机访问(如 operator[]),元素访问时间复杂度为 O (N)。
    • 内存碎片化问题:动态节点分配可能导致缓存利用率低。
典型应用场景
  • 频繁插入 / 删除操作的场景(如链表结构的动态数据管理)。
  • 无需随机访问,注重数据动态调整的场景(如任务队列、事件链表)。

1.2 与数组(vector)的区别

特性list(链表)vector(数组)
内存分布非连续(动态节点)连续(一块内存)
随机访问慢(需逐个遍历)快(下标直接访问)
插入 / 删除效率快(仅修改指针)慢(需移动元素)
典型用法队列、链表结构数组运算、随机访问场景

二、List 接口详解与实践

2.1 构造函数

构造方式代码示例说明
默认构造list<int> l1;空链表
初始化数量与值list<int> l1(5, 10);5 个值为 10 的元素
区间构造list<int> l1(arr, arr+5);用数组 arr 的前 5 个元素初始化
拷贝构造list<int> l2(l1);复制已有链表 l1
2.1.1 头文件与命名空间
#include <list>    // list 头文件
using namespace std; // 使用 std 命名空间
2.1.2 创建 list
(1)空 list
list<int> numbers; // 空链表,存储 int 类型数据
(2)初始化指定元素
  • 指定数量和值
    list<int> l1(5, 10); // 5个元素,每个值都是 10
    
  • 用数组 / 其他容器初始化
    int arr[] = {1, 2, 3};
    list<int> l2(arr, arr + 3); // 用数组前3个元素初始化
    
  • 拷贝其他 list
    list<int> l3(l2); // 复制 l2 的所有元素
    

2.2 迭代器操作

  • 正向迭代器begin() 指向首元素,end() 指向尾后节点(哨兵位)。
  • 反向迭代器rbegin() 指向尾元素,rend() 指向头前节点(哨兵位)。
  • 特性
    • 正向迭代器 ++ 向后移动,反向迭代器 ++ 向前移动。
    • 支持 !=== 比较。

代码演示(遍历元素)

list<int> l = {1, 3, 5, 7, 9};
// 正向遍历
for (auto it = l.begin(); it != l.end(); ++it) {cout << *it << " "; // 输出:1 3 5 7 9
}
// 反向遍历
for (auto rit = l.rbegin(); rit != l.rend(); ++rit) {cout << *rit << " "; // 输出:9 7 5 3 1
}

2.3 容量与元素访问

接口功能返回值
empty()判断链表是否为空bool
size()返回有效节点数size_type
front() 返回首元素引用T&
back()返回尾元素引用T&
 2.3.1检查空链表与获取长度
list<int> l;
if (l.empty()) { // 判断是否为空cout << "链表为空!";
}
l.push_back(1);
cout << "元素个数:" << l.size(); // 输出:1
 2.3.2获取首尾元素
list<int> l = {1, 3, 5};
cout << "首元素:" << l.front(); // 输出:1
cout << "尾元素:" << l.back();  // 输出:5

注意:避免在空链表上调用 front()/back(),可能引发未定义行为。

2.4 增删改操作

接口功能描述时间复杂度
push_front(val)头部插入元素O(1)
push_back(val)尾部插入元素O(1)
pop_front()头部删除元素O(1)
pop_back()尾部删除元素O(1)
insert(pos, val)在 pos 位置插入元素O(1)
erase(pos)删除 pos 位置的元素O(1)
clear()清空所有元素O(N)

关键细节

  • 插入操作insert 返回新插入元素的迭代器,可用于连续插入。
  • 删除操作:删除后,指向被删节点的迭代器失效,其他迭代器不受影响。
2.4.1 向 list 中添加元素
 (1)头部插入(快!O (1))
list<int> l;
l.push_front(1); // 头部插入 1 → list: [1]
l.push_front(0); // 头部插入 0 → list: [0, 1]
(2)尾部插入(快!O (1))
l.push_back(2); // 尾部插入 2 → list: [0, 1, 2]
(3)任意位置插入(需迭代器定位)
auto it = l.begin(); // 指向首元素(0)的迭代器
l.insert(it, 99); // 在 it 位置插入 99 → list: [99, 0, 1, 2]
 2.4.2. 从 list 中删除元素
(1)头部删除(快!O (1))
l.pop_front(); // 删除头部元素 99 → list: [0, 1, 2]
(2)尾部删除(快!O (1))
l.pop_back(); // 删除尾部元素 2 → list: [0, 1]
(3)删除指定位置元素
auto it = l.begin(); // it 指向 0
++it; // it 指向 1
l.erase(it); // 删除 1 → list: [0]
(4)删除指定值的元素(需遍历查找)
list<int> l = {1, 2, 3, 2, 4};
l.remove(2); // 删除所有值为 2 的元素 → list: [1, 3, 4]

代码演示(删除偶数元素)

list<int> l = {1, 2, 3, 4, 5};
auto it = l.begin();
while (it != l.end()) {if (*it % 2 == 0) {it = l.erase(it); // erase返回下一个有效迭代器} else {++it;}
}
// 结果:l = {1, 3, 5}

2.5 迭代器失效问题

  • 插入场景:不会导致迭代器失效(链表节点独立,插入不影响其他节点指针)。
  • 删除场景
    • 被删除节点的迭代器失效。
    • 正确做法:使用 erase(it++) 或 it = erase(it) 更新迭代器。

错误示例

  • 场景:删除元素后,指向被删节点的迭代器会失效,需及时更新。
  • 错误代码
    list<int> l = {1, 2, 3};
    auto it = l.begin();
    while (it != l.end()) {if (*it == 2) {l.erase(it); // 错误!it 失效,下次循环非法访问}++it; // 这里会访问失效的迭代器
    }
    
  • 正确代码
    while (it != l.end()) {if (*it == 2) {it = l.erase(it); // 用 erase 返回的下一个迭代器更新 it} else {++it;}
    }
    

三、List 模拟实现关键点

3.1 节点结构定义

template <class T>
struct ListNode {T val;ListNode* prev;ListNode* next;ListNode(const T& x) : val(x), prev(nullptr), next(nullptr) {}
};

3.2 正向迭代器实现

  • 核心成员:封装节点指针 ListNode<T>* node
  • 操作重载
    • operator*():返回节点值(return node->data;)。
    • operator++():前置递增,指向 next 节点(node = node->next;)。
    • operator--():前置递减,指向 prev 节点(node = node->prev;)。
struct list_iterator
{typedef list_node<T> Node;typedef list_iterator<T,Ref,ptr> Self;Node* _pnode;list_iterator(Node* val):_pnode(val){ }Ref operator*(){return _pnode->_data;}ptr operator->(){return &_pnode->_data;}Self& operator++(){_pnode = _pnode->_next;return *this;}Self operator++(int){Self tmp(*this);_pnode = _pnode->_next;return tmp;}Self& operator--(){_pnode = _pnode->_prev;return *this;}		Self operator--(int){Self tmp(*this);_pnode = _pnode->_prev;return tmp;}bool operator!=(const Self& s){return _pnode != s._pnode;}bool operator==(const Self& s){return _pnode == s._pnode;}
};

四、List 与 Vector 对比

特性VectorList关键差异原因
底层结构连续内存(动态数组)非连续内存(双向链表)内存布局决定访问与修改效率
随机访问O (1)(支持 []O (N)(需遍历链表)数组下标 vs 指针遍历
插入 / 删除平均 O (N)(移动元素)O (1)(修改指针)数组元素搬移 vs 链表指针操作
内存效率高(缓存友好)低(节点指针开销)连续内存 vs 碎片化内存
迭代器类型原生指针(T*封装迭代器(含指针逻辑)链表需要封装指针操作
迭代器失效插入可能全量失效(扩容)仅删除节点失效数组扩容导致指针失效 vs 链表节点独立
典型场景数据频繁访问(如数组运算)数据频繁增删(如消息队列)根据操作模式选择容器

五、常见错误

 5.1. 不支持随机访问(与数组的区别)

  • 错误用法
    list<int> l = {1, 2, 3};
    cout << l[1]; // 错误!list 没有 [] 运算符
    
  • 正确做法:用迭代器遍历或 front()/back() 获取首尾元素。

5.2. 空链表操作风险

  • 错误用法
    list<int> l;
    cout << l.front(); // 空链表调用 front(),程序可能崩溃
    
  • 正确做法:先检查 empty()
    if (!l.empty()) {cout << l.front();
    }
    

六、总结

list 是 STL 中基于双向循环链表的容器,适合频繁增删、无需随机访问的场景。其核心优势在于 O (1) 时间复杂度的插入 / 删除操作,以及双向遍历能力,但代价是较高的内存开销和较低的缓存利用率。学习 list 时,需重点掌握:

  1. 迭代器的双向操作与失效规则(尤其是删除后的迭代器更新)。
  2. 与 vector 的对比,根据实际需求选择容器。
  3. 模拟实现时双向链表与迭代器封装的逻辑。

附录

模拟实现

//stl_list.h
#pragma once
#include<iostream>
#include<assert.h>namespace my_list
{template<class T>struct list_node{list_node* _prev;list_node* _next;T _data;list_node(const T& val=T()):_data(val),_prev(nullptr),_next(nullptr){ }};template<class T,class Ref,class ptr>struct list_iterator{typedef list_node<T> Node;typedef list_iterator<T,Ref,ptr> Self;Node* _pnode;list_iterator(Node* val):_pnode(val){ }Ref operator*(){return _pnode->_data;}ptr operator->(){return &_pnode->_data;}Self& operator++(){_pnode = _pnode->_next;return *this;}Self operator++(int){Self tmp(*this);_pnode = _pnode->_next;return tmp;}Self& operator--(){_pnode = _pnode->_prev;return *this;}		Self operator--(int){Self tmp(*this);_pnode = _pnode->_prev;return tmp;}bool operator!=(const Self& s){return _pnode != s._pnode;}bool operator==(const Self& s){return _pnode == s._pnode;}};template<class T>class list{typedef list_node<T> Node;public:typedef list_iterator<T,T&,T*> iterator;typedef list_iterator<T,const T&,const T*> const_iterator;iterator begin(){return iterator(_head->_next);}		const_iterator begin()const{return const_iterator(_head->_next);}iterator end(){return iterator(_head);}		const_iterator end()const{return const_iterator(_head);}void empty_init(){_head = new Node;_head->_prev = _head;_head->_next = _head;_size = 0;}list(){empty_init();}list(std::initializer_list<T> il){empty_init();for (auto e : il){push_back(e);}}list(const list<T>& ll){empty_init();for (auto e : ll){push_back(e);}}list<T>& operator=(list<T> ll){swap(ll);return *this;}~list(){clear();delete _head;_head = nullptr;}size_t size(){return _size;}void swap(list<T>& ll){std::swap(_head, ll._head);std::swap(_size, ll._size);}void push_back(const T& val){/*Node* tmp = new Node(val);tmp->_prev = _head->_prev;tmp->_next = _head;_head->_prev->_next = tmp;_head->_prev = tmp;_size++;*/insert(end(), val);}void push_head(const T& val){insert(begin(), val);}void pop_back(){erase(--end());}void pop_head(){erase(begin());}void clear(){iterator it = begin();while (it != end()){it = erase(it);}}void insert(iterator pos, const T& val){Node* newnode = new Node(val);Node* tmp = pos._pnode;newnode->_next = tmp;newnode->_prev = tmp->_prev;tmp->_prev->_next = newnode;tmp->_prev = newnode;++_size;}iterator erase(iterator pos){assert(pos != end());Node* cur = pos._pnode;Node* tmp = cur->_next;cur->_prev->_next = cur->_next;cur->_next->_prev = cur->_prev;delete cur;_size--;return tmp;}private:Node* _head;size_t _size;};
}

相关文章:

C++ STL 容器:List 深度解析与实践指南

一、List 容器概述 1.1底层结构与特性 数据结构&#xff1a;双向循环链表&#xff08;带哨兵位头结点&#xff09;&#xff0c;每个节点包含前驱指针、后继指针和数据域。核心优势&#xff1a; 高效插入 / 删除&#xff1a;任意位置操作时间复杂度为 O (1)&#xff0c;无需移…...

每天掌握一个Linux命令 - ab(Apache Benchmark)

Linux 命令工具 ab 使用指南 一、工具概述 ab&#xff08;Apache Benchmark&#xff09; 是 Apache 官方提供的开源压力测试工具&#xff0c;用于衡量 Web 服务器的性能。它通过模拟多并发请求&#xff0c;测试服务器在高负载下的响应速度、吞吐量和稳定性&#xff0c;常用于…...

与 PyCharm 官方沟通解决开发环境问题记录(进展:官方已推出2个新的修复版本)

​​​​​​主题&#xff1a;有关 PyCharm 中终端和环境激活问题的反馈&#xff1a;PY-81233 前言 目前进展&#xff1a; 官方已有2个修复版本推出测试。 更新方法&#xff1a; 使用JetBrains Toolbox App&#xff0c;如下图所示&#xff0c;从“其他版本”进入查看更新。…...

Python的分布式网络爬虫系统实现

1. 系统架构概述 一个典型的分布式网络爬虫系统通常包含以下几个核心组件&#xff1a; 1.主节点&#xff08;Master Node&#xff09;&#xff1a; 任务调度&#xff1a;负责将抓取任务分配给各个工作节点。URL 管理&#xff1a;维护待抓取的 URL 队列和已抓取的 URL 集合&a…...

Vue快速上手(业务、技术、报错)

Vue 技术业务报错 技术 业务 Vueelement-ui&#xff0c;实现表格渲染缩略图&#xff0c;鼠标悬浮缩略图放大&#xff0c;点击缩略图播放视频&#xff08;一&#xff09; 报错 vue修改配置文件.env.development不生效 vue前端downloadFile报错&#xff1a;Error parsing HT…...

taro + vue3 实现小程序sse长连接实时对话

前言 taro.request是可以实现sse长连接的&#xff0c;但是呢其中有俩大坑&#xff0c;找了许多资料也没解决&#xff0c;后续解决办法也与后端商量改用WebSocket来实现。 代码实现 SSEManager.js: import { getAccessToken } from "../xx/xx"; import { TextDecode…...

使用MATLAB求解微分方程:从基础到实践

使用MATLAB求解微分方程&#xff1a;从基础到实践 微分方程是描述自然界和工程领域中许多现象的重要数学工具。MATLAB提供了强大的工具来求解各种类型的微分方程。本文将介绍如何使用MATLAB求解常微分方程(ODE)。 1. 基本ODE求解器 MATLAB提供了多种ODE求解器&#xff0c;最…...

基于MATLAB的大规模MIMO信道仿真

1. 系统模型与参数设置 以下是一个单小区大规模MIMO系统的参数配置示例&#xff0c;适用于多发多收和单发单收场景。 % 参数配置 params.N_cell 1; % 小区数量&#xff08;单小区仿真&#xff09; params.cell_radius 500; % 小区半径&#xff08;米&#xff09…...

如何在 Windows 和 Mac 上擦拭和清洁希捷外置硬盘

希捷外置硬盘广泛用于存储目的&#xff0c;但有时您可能出于多种目的需要擦除或清洁希捷外置硬盘&#xff0c;例如转售、重复使用、捐赠等。为了释放硬盘上的存储空间或确保没有人可以从硬盘中恢复您的信息&#xff0c;擦除硬盘是必要的步骤。无论您使用的是 Windows 还是 Mac&…...

Vue 3.0 中状态管理Vuex 与 Pinia 的区别

在 Vue.js 应用开发中&#xff0c;状态管理是构建复杂应用的关键环节。随着 Vue 3 的普及和 Composition API 的引入&#xff0c;开发者面临着状态管理库的选择问题&#xff1a;是继续使用经典的 Vuex&#xff0c;还是转向新兴的 Pinia&#xff1f;本文将从设计理念、API 设计、…...

第三届黄河流域网安技能挑战赛复现

Web 奶龙牌图片处理器2.0 这题&#xff0c;之前只了解过 .user.ini 文件&#xff0c;并为遇到实操题 但赛前差点就做到下面这题了&#xff0c;不多说&#xff0c;复现之前先看看下面这题 靶场&#xff1a; 攻防世界 没错&#xff0c;又做上文件上传题了&#xff0c;别看…...

python 生成复杂表格,自动分页等功能

py&#xff54;&#xff48;&#xff4f;&#xff4e; 生成复杂表格&#xff0c;自动分页等功能 解决将Python中的树形目录数据转换为Word表格&#xff0c;并生成带有合并单元格的检测报告的问题。首先&#xff0c;要解决“tree目录数据”和“Word表格互换”&#xff0c;指将树…...

2025年高防IP与游戏盾深度对比:如何选择最佳防护方案?

2025年&#xff0c;随着DDoS攻击规模的指数级增长和混合攻击的常态化&#xff0c;高防IP与游戏盾成为企业网络安全的核心选择。然而&#xff0c;两者在功能定位、技术实现及适用场景上存在显著差异。本文结合最新行业实践与技术趋势&#xff0c;全面解析两者的优劣&#xff0c;…...

在 Vue + Vite 项目中,直接使用相对路径或绝对路径引用本地图片资源时,图片无法正确显示。

Vue 项目中静态资源引用问题 1.问题描述 在 Vue Vite 项目中&#xff0c;直接使用相对路径或绝对路径引用本地图片资源时&#xff0c;图片无法正确显示。 错误示例 javascript // 错误方式1&#xff1a;使用相对路径 const products [ { name: iPhone 14 Pro, image: .…...

判断手机屏幕上的横向滑动(左滑和右滑)

在JavaScript中&#xff0c;你可以通过监听触摸事件&#xff08;touch events&#xff09;来判断用户在手机屏幕上的横向滑动方向。以下是实现方法&#xff1a; 基本实现方案 let touchStartX 0; let touchEndX 0;function handleTouchStart(event) {touchStartX event.ch…...

用户有一个Django模型没有设置主键,现在需要设置主键。

用户有一个Django模型没有设置主键&#xff0c;现在需要设置主键。 from django.db import modelsclass CategoryAssistentModel(models.Model):second_level_category models.CharField(max_length100, nullTrue, blankTrue)third_level_category models.CharField(max_len…...

【文献阅读】EndoChat: Grounded Multimodal Large Language Model for Endoscopic Surgery

[2501.11347] EndoChat: Grounded Multimodal Large Language Model for Endoscopic Surgery 2025年1月 数据可用性 Surg-396K 数据集可在 GitHub - gkw0010/EndoChat 公开获取。 代码可用性 EndoChat 的代码可在 GitHub - gkw0010/EndoChat 下载。 摘要 近年来&#xff…...

React JSX语法介绍(JS XML)(一种JS语法扩展,允许在JS代码中编写类似HTML的标记语言)Babel编译

在线调试网站&#xff1a;https://zh-hans.react.dev/learn 文章目录 JSX&#xff1a;现代前端开发的声明式语法概述JSX的本质与工作原理什么是JSXJSX转换流程 JSX语法特性表达式嵌入&#xff08;JSX允许在大括号内嵌入任何有效的JavaScript表达式&#xff09;属性传递&#xf…...

【R语言编程绘图-箱线图】

基本箱线图绘制 使用ggplot2绘制箱线图的核心函数是geom_boxplot()。以下是一个基础示例&#xff0c;展示如何用iris数据集绘制不同物种&#xff08;Species&#xff09;的萼片长度&#xff08;Sepal.Length&#xff09;分布&#xff1a; library(ggplot2) ggplot(iris, aes(…...

【elasticsearch 7 或8 的安装及配置SSL 操作指引】

1.标题获取安装文件 cd /opt/tools wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-8.11.4-linux-x86_64.tar.gz tar -zxvf elasticsearch-8.11.4-linux-x86_64.tar.gz mv /opt/tools/elasticsearch-8.11.4 /opt/elasticsearch #配置vm.max_map_co…...

GitHub 趋势日报 (2025年05月23日)

本日报由 TrendForge 系统生成 https://trendforge.devlive.org/ &#x1f310; 本日报中的项目描述已自动翻译为中文 &#x1f4c8; 今日整体趋势 Top 10 排名项目名称项目描述今日获星总星数语言1All-Hands-AI/OpenHands&#x1f64c;开放式&#xff1a;少代码&#xff0c;做…...

MongoDB索引:原理、实践与优化指南

为什么索引对数据库如此重要&#xff1f; 在现代应用开发中&#xff0c;数据库性能往往是决定用户体验的关键因素。想象一下&#xff0c;当你在电商平台搜索商品时&#xff0c;如果每次搜索都需要等待5-10秒才能看到结果&#xff0c;这种体验是多么令人沮丧。MongoDB作为最流行…...

SQL实战之索引优化(单表、双表、三表、索引失效)

文章目录 单表优化双表优化三表优化结论索引失效 单表优化 总体原则&#xff1a;建立索引并合理使用&#xff0c;避免索引失效 案例说明&#xff1a;查询category_ id 为1且comments大于1的情况下,views最多的article_ id: 传统方案&#xff1a; explain select id, author_ id…...

[7-1] ADC模数转换器 江协科技学习笔记(14个知识点)

1 2 3 4 5 6 7 8 9 10 11 12 13 14 DMA&#xff08;Direct Memory Access&#xff0c;直接内存访问&#xff09;是一种硬件特性&#xff0c;它允许某些硬件子系统直接访问系统的内存&#xff0c;而无需CPU的介入。这样&#xff0c;CPU就可以处理其他任务&#xff0c;从而提高系…...

SSM整合:Spring+SpringMVC+MyBatis完美融合实战指南

前言 在Java企业级开发领域&#xff0c;SSM&#xff08;SpringSpringMVCMyBatis&#xff09;框架组合一直占据着重要地位。这三个轻量级框架各司其职又相互配合&#xff0c;为开发者提供了高效、灵活的开发体验。本文将深入探讨SSM框架的整合过程&#xff0c;揭示整合背后的原…...

Spring Boot分页查询进阶:整合Spring Data REST实现高效数据导航

目录&#xff1a; 引言分页查询基础回顾 2.1 Spring Data JPA分页接口 2.2 Pageable与Page的使用 2.3 常见分页参数设计Spring Data REST简介 3.1 HATEOAS与超媒体驱动API 3.2 Spring Data REST核心功能 3.3 自动暴露Repository接口整合Spring Boot与Spring Data REST 4.1 项目…...

阿里云 Serverless 助力海牙湾构建弹性、高效、智能的 AI 数字化平台

作者&#xff1a;赵世振、十眠、修省 “通过阿里云 Serverless 架构&#xff0c;我们成功解决了弹性能力不足、资源浪费与运维低效的痛点。SAE 的全托管特性大幅降低技术复杂度。未来&#xff0c;我们将进一步探索 Serverless 与 AI 的结合&#xff0c;为客户提供更智能的数字…...

升级node@22后运行npm install报错 distutils not found

从node20升级到node22后&#xff0c;在运行 npm install 的时候报了很多 gyp 错误&#xff0c;其中包括 npm error npm error ModuleNotFoundError: No module named distutils。 问题原因是我在使用 brew install node22 的过程中自动把 python 升级到了 3.13。而 distutils …...

一个开源的多播放源自动采集在线影视网站

这里写自定义目录标题 欢迎使用Markdown编辑器GoFilm简介项目部署1、前置环境准备1.2 redis 配置 film-api 后端服务配置将 GoFilm 项目根目录下的 film 文件夹上传到 linux 服务器的 /opt 目录下 2. 构建运行1. docker 部署1.1 安装 docker , docker compose 环境 注意事项: 2…...

【PhysUnits】10 减一操作(sub1.rs)

一、源码 代码实现了一个类型级别的减一操作(Sub1 trait)&#xff0c;通过Rust的类型系统在编译期完成数值减一的计算。 //! 减一操作特质实现 / Decrement operation trait implementation //! //! 提供类型级别的减一计算 / Provides type-level decrement operationuse su…...