【C++指南】C++ list容器完全解读(二):list模拟实现,底层架构揭秘
.
💓 博客主页:倔强的石头的CSDN主页
📝Gitee主页:倔强的石头的gitee主页
⏩ 文章专栏:《C++指南》
期待您的关注![]()
文章目录
- 引言
- 一、链表节点设计:双向链表的基石
- 1.1 节点类的实现
- 二、list框架与核心成员函数
- 2.1 list类的成员变量
- 2.2 构造函数与初始化
- 2.3 深拷贝
- 2.4 赋值运算符重载:传统写法 vs 现代写法
- 2.4 构造函数重载与冲突解决
- 三、修改操作:插入与删除
- 3.1 插入操作`insert`
- 3.2 删除操作`erase`
- 四、其他关键函数实现
- 4.1 容量操作
- 4.1 交换函数
- 4.2 自定义swap的高效实现
- 结语
引言
在上一篇文章【C++指南】STL list容器完全解读(一):从入门到掌握基础操作中,我们深入探讨了list容器的核心特性、使用场景及接口规范。
本文作为系列第二篇,将聚焦于list的底层模拟实现,通过手写双向链表结构,揭示其高效插入删除的底层逻辑。
通过本文,您将掌握:
- 双向链表的节点设计与内存管理
- list核心成员函数的实现原理
- 深拷贝与现代C++优化技巧
- STL容器设计中的关键思想
一、链表节点设计:双向链表的基石
1.1 节点类的实现
list_node
是链表的原子单位,需包含数据域和前后指针:
template <class T>
struct list_node {T data; // 数据域list_node<T>* next; // 后继指针list_node<T>* prev; // 前驱指针// 构造函数:支持默认值初始化list_node(const T& x = T()) : data(x), next(nullptr), prev(nullptr) {}
};
关键点:
- 模板化设计支持任意数据类型
- 默认构造函数初始化指针为
nullptr
,避免野指针
二、list框架与核心成员函数
2.1 list类的成员变量
template <class T>
class list {
private:typedef list_node<T> Node;Node* _head; // 头节点(哨兵节点)size_t _size; // 元素数量
public:// 迭代器声明(下篇详解)typedef list_iterator<T, T&, T*> iterator;// ... 其他成员函数
};
核心设计思想:
- 头节点作为哨兵节点,使空链表的
begin()
和end()
统一指向_head
_size
记录元素数量,避免遍历统计
2.2 构造函数与初始化
// 默认构造:创建空链表
list() { empty_init(); }// 初始化函数:构建头节点闭环
void empty_init() {_head = new Node();_head->next = _head;_head->prev = _head;_size = 0;
}
注意事项:
- 头节点的
next
和prev
均指向自身,形成环形结构
2.3 深拷贝
拷贝构造:
list(const list<T>& s) {empty_init();for (auto& i : s) push_back(i); // 深拷贝
}
2.4 赋值运算符重载:传统写法 vs 现代写法
传统写法(显式深拷贝)
list<T>& operator=(const list<T>& s) { if (this != &s) { // 防止自赋值 clear(); // 清空当前链表 for (auto& val : s) { push_back(val); // 逐元素深拷贝 } } return *this;
}
缺点:代码冗余,需手动处理资源释放与拷贝。
现代写法(资源交换)
list<T>& operator=(list<T> s) { swap(s); // 传递临时对象,利用拷贝构造完成深拷贝 return *this;
}
优势:
- 利用拷贝构造函数生成临时对象
s
,自动完成深拷贝 - 通过
swap
交换资源,临时对象s
析构时自动释放旧数据
2.4 构造函数重载与冲突解决
1. 多个val值的构造
// 填充构造函数:创建n个值为val的元素
list(size_t n, const T& val = T()) { empty_init(); for (size_t i=0; i<n; ++i) push_back(val);
} // 重载int版本,避免与迭代器范围构造冲突
list(int n, const T& val = T()) { empty_init(); for (int i=0; i<n; ++i) push_back(val);
}
2. 初始化列表构造
list(initializer_list<T> il) { empty_init(); for (auto& elem : il) push_back(elem);
}
3. 迭代器范围构造
template <class InputIterator>
list(InputIterator first, InputIterator last) { empty_init(); while (first != last) { push_back(*first); ++first; }
}
为什么需要重载int n
版本?
若只有模板版本的迭代器范围构造,当用户调用list<int> lst(5, 1)
时:
如下图所示,错误的调用了为迭代器准备的模板函数,就会对int解引用
- 编译器优先匹配
InputIterator
版本(int
被推导为迭代器类型) - 导致逻辑错误(试图对整数
5
和1
解引用)
解决方案:提供int n
的重载版本,明确匹配数值构造场景。
三、修改操作:插入与删除
3.1 插入操作insert
iterator insert(iterator pos, const T& val) {Node* cur = pos._node; // 当前节点Node* prev = cur->prev; // 前驱节点Node* new_node = new Node(val); // 新节点// 链接新节点prev->next = new_node;new_node->prev = prev;new_node->next = cur;cur->prev = new_node;_size++;return iterator(new_node); // 返回新节点迭代器
}
复用插入实现push_back/push_front
:
void push_back(const T& x) { insert(end(), x); }
void push_front(const T& x) { insert(begin(), x); }
3.2 删除操作erase
iterator erase(iterator pos) {assert(pos != end()); // 禁止删除头节点Node* cur = pos._node;Node* prev = cur->prev;Node* next = cur->next;// 跳过被删节点prev->next = next;next->prev = prev;delete cur;_size--;return iterator(next); // 返回下一节点迭代器
}
复用删除实现pop_back/pop_front
:
void pop_back() { erase(--end()); }
void pop_front() { erase(begin()); }
四、其他关键函数实现
4.1 容量操作
// 清空链表(保留头节点)
void clear() {iterator it = begin();while (it != end()) it = erase(it);
}bool empty()//判空
{return _size == 0;
}size_t size()//获取链表元素个数
{return _size;
}
4.1 交换函数
STL的std::swap
通过三次拷贝完成交换:
template <class T>
void swap(T& a, T& b) { T tmp(a); a = b; b = tmp;
}
问题:
- 对链表而言,逐节点拷贝效率极低(时间复杂度O(n))
4.2 自定义swap的高效实现
类内swap:直接交换头指针与_size
void swap(list<T>& other) { std::swap(_head, other._head); // O(1)交换 std::swap(_size, other._size);
}
全局swap适配:确保ADL正确调用
template <class T>
void swap(list<T>& a, list<T>& b) { a.swap(b); // 调用类内swap
}
为何需要全局swap?
- 若用户调用
swap(lst1, lst2)
,编译器优先查找参数关联的命名空间 - 全局swap确保调用自定义实现,而非低效的
std::swap
结语
本文从双向链表的节点设计出发,逐步实现了list的核心功能,揭示了STL容器设计中的内存管理与接口复用思想。
在下一篇文章中,我们将深入探讨迭代器的封装与类型萃取技术
下篇预告:《【C++指南】C++ list容器完全解读(三):list迭代器的实现与优化》—— 揭秘STL迭代器如何实现“透明访问”与高效遍历!
相关文章:

【C++指南】C++ list容器完全解读(二):list模拟实现,底层架构揭秘
. 💓 博客主页:倔强的石头的CSDN主页 📝Gitee主页:倔强的石头的gitee主页 ⏩ 文章专栏:《C指南》 期待您的关注 文章目录 引言一、链表节点设计:双向链表的基石1.1 节点类的实现 二、list框架与核心成员函…...

[神经网络]使用olivettiface数据集进行训练并优化,观察对比loss结果
结合归一化和正则化来优化网络模型结构,观察对比loss结果 搭建的神经网络,使用olivettiface数据集进行训练,结合归一化和正则化来优化网络模型结构,观察对比loss结果 from sklearn.datasets import fetch_olivetti_faces #倒入数…...
小明的Java面试奇遇之智能家装平台架构设计与JVM调优实战
一、文章标题 小明的Java面试奇遇之智能家装平台架构设计与JVM调优实战 二、文章标签 Java面试, 智能家装, 微服务架构, 高并发设计, JVM调优, SpringCloud, 消息队列, 分布式缓存, 架构设计, 面试技巧 三、文章概述 本文模拟了程序员小明应聘智能家装平台后端架构师的5轮…...
n8n:技术团队的智能工作流自动化助手
在当前数字化时代,自动化已经成为提高效率和减轻人工工作负担的一大推动力。今天,我们要为大家介绍一款极具潜力的开源项目——n8n,它不仅拥有广泛的应用场景,还具备内置AI功能,能够完全满足技术团队的高效工作需求。n8n的出现,为技术团队提供了自由编程与快速自动化构建…...
Flink 核心机制与源码剖析系列
Flink 核心机制与源码剖析系列 目录 第一篇:Flink 状态管理原理与源码深度剖析第二篇:水位线、事件时间与定时器源码全流程第三篇:Flink CEP 模式建模与高效事件匹配机制 第一篇:Flink 状态管理原理与源码深度剖析 1. 背景与意…...

华院计算出席信创论坛,分享AI教育创新实践并与燧原科技共同推出教育一体机
5月21日,信创论坛于上海漕河泾会议中心举办。本次论坛以“聚力融合,繁荣生态”为主题,话题聚焦工业制造、交通运输、金融、教育、医疗等领域。华院计算技术(上海)股份有限公司(以下简称“华院计算”&#x…...

华为OD机试真题——会议接待 /代表团坐车(2025A卷:200分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
2025 A卷 200分 题型 本文涵盖详细的问题分析、解题思路、代码实现、代码详解、测试用例以及综合分析; 并提供Java、python、JavaScript、C++、C语言、GO六种语言的最佳实现方式! 本文收录于专栏:《2025华为OD真题目录+全流程解析/备考攻略/经验分享》 华为OD机试真题《会议…...

LabVIEW Val (Sgnl) 属性
在 LabVIEW 事件驱动架构中,Val (Sgnl) 属性(Value (Signaling))是实现编程触发与用户交互行为一致性的关键技术。与普通 Value 属性不同,Val (Sgnl) 在修改控件值的同时强制生成值改变事件,确保程序逻辑与 UI 交互保持…...

STM32G4 电机外设篇(三) TIM1 发波 和 ADC COMP DAC级联
目录 一、STM32G4 电机外设篇(三) TIM1 发波 和 ADC COMP DAC级联1 TIM1 高级定时器发波1.1 stm32cubemx配置 2 TIM1 ADC COMP DAC级联2.1 stm32cubemx配置 附学习参考网址欢迎大家有问题评论交流 (* ^ ω ^) 一、STM32G4 电机外设篇(三&…...

DAY 35 超大力王爱学Python
知识点回顾: 三种不同的模型可视化方法:推荐torchinfo打印summary权重分布可视化进度条功能:手动和自动写法,让打印结果更加美观推理的写法:评估模式 作业:调整模型定义时的超参数,对比下效果。…...

【数据结构】图的存储(十字链表)
弧节点 tailvex数据域:存储弧尾一端顶点在顺序表中的位置下标;headvex 数据域:存储弧头一端顶点在顺序表中的位置下标;hlink 指针域:指向下一个以当前顶点作为弧头的弧;tlink 指针域:指向下一个…...
005 flutter基础,初始文件讲解(4)
书接上回,今天继续完成最后的讲解: class _MyHomePageState extends State<MyHomePage> {int _counter 0;void _incrementCounter() {setState(() {_counter;});}可以看到,这里的_MyHomePageState是一个类,继承于 State&l…...

Redis最佳实践——秒杀系统设计详解
基于Redis的高并发秒杀系统设计(十万级QPS) 一、秒杀系统核心挑战 瞬时流量洪峰:100万 QPS请求冲击库存超卖风险:精准扣减防止超卖系统高可用性:99.99%服务可用性要求数据强一致性:库存/订单/支付状态同步…...

STM32软件spi和硬件spi
核心观点 本文主要介绍了SPI通信的两种实现方式:软件SPI和硬件SPI。详细阐述了SPI通信协议的基本概念、硬件电路连接方式、移位示意图、时序基本单元以及四种工作模式。同时,对W25Q64模块进行了详细介绍,包括其硬件电路、框图以及操作注意事…...
MATLAB实战:人脸检测与识别实现方案
我们要用电脑识别照片或视频中的人脸,并知道是谁的脸。就像手机相册能自动识别照片里的人是谁一样。 🔍 人脸检测(找脸) 目标:在图片中找到人脸的位置 怎么做: 用MATLAB的"人脸扫描仪"ÿ…...

深度刨析树结构(从入门到入土讲解AVL树及红黑树的奥秘)
目录 树的表示 二叉树的概念及结构(重点学习) 概念 : 特点: 树与非树 特殊的二叉树 二叉树的性质(重点) 二叉树的存储结构 堆的概念及结构 建堆方式: 向下调整算法 向上调整算法 建堆第一步初始化 建…...

【Linux】shell的条件判断
目录 一.使用逻辑运算符判定命令执行结果 二.条件判断方法 三.判断表达式 3.1文件判断表达式 3.2字符串测试表达式 3.3整数测试表达式 3.4逻辑操作符 一.使用逻辑运算符判定命令执行结果 && 在命令执行后如果没有任何报错时会执行符号后面的动作|| 在命令执行后…...

第九天:java注解
注解 1 什么是注解(Annotation) public class Test01 extends Object{//Override重写的注解Overridepublic String toString() {return "Test01{}";} }2 内置注解 2.1 Override Override重写的注解 Override public String toString() {ret…...

十一、【核心功能篇】测试用例管理:设计用例新增编辑界面
【核心功能篇】测试用例管理:设计用例新增&编辑界面 前言准备工作第一步:创建测试用例相关的 API 服务 (src/api/testcase.ts)第二步:创建测试用例编辑页面组件 (src/views/testcase/TestCaseEditView.vue)第三步:配置测试用例…...
react-native的token认证流程
在 React Native 中实现 Token 认证是移动应用开发中的常见需求,它用于验证用户的身份并授权其访问受保护的 API 资源。 Token 认证的核心流程: 用户登录 (Login): 用户在前端输入用户名和密码。前端将这些凭据发送到后端 API。后端验证凭据。如果验证成…...
ERP系统中商品定价功能设计:支持渠道、会员与批发场景的灵活定价机制
在现代零售、批发与电商环境下,商品的定价策略日益复杂。一个优秀的ERP系统不仅需要管理商品基础信息、库存与订单,还必须提供一套灵活且可扩展的商品定价机制,以满足: 不同销售渠道(如线上平台、线下门店、分销商&…...

Spring是如何实现属性占位符解析
Spring属性占位符解析 核心实现思路1️⃣ 定义占位符处理器类2️⃣ 处理 BeanDefinition 中的属性3️⃣ 替换具体的占位符4️⃣ 加载配置文件5️⃣ Getter / Setter 方法 源码见:mini-spring 在使用 Spring 框架开发过程中,为了实现配置的灵活性…...
数据结构之ArrayList
系列文章目录 目录 系列文章目录 前言 一、数据结构的前置语法 1. 时空复杂度 2. 包装类 3. 泛型 二、ArrayList 和顺序表 1. 顺序表的模拟实现 2. 源码 3. ArrayList 的优缺点 前言 本文介绍数据结构的前置算法,以及 ArrayList 的模拟实现,部…...

DDR4读写压力测试
1.1测试环境 1.1.1整体环境介绍 板卡: pcie-403板卡 主控芯片: Xilinx xcvu13p-fhgb2104-2 调试软件: Vivado 2018.3 代码环境: Vscode utf-8 测试工程: pcie403_user_top 1.1.2硬件介绍 UD PCIe-403…...
uniapp 开发企业微信小程序时,如何在当前页面真正销毁前或者关闭小程序前调用一个api接口
在 UniApp 开发企业微信小程序时,若需在页面销毁或小程序关闭前调用 API 接口,需结合页面生命周期和应用生命周期实现。以下是具体实现方案及注意事项: 一、在页面销毁前调用 API(页面级) 通过页面生命周期钩子 onUnl…...
WPF 按钮点击音效实现
WPF 按钮点击音效实现 下面我将为您提供一个完整的 WPF 按钮点击音效实现方案,包含多种实现方式和高级功能: 完整实现方案 MainWindow.xaml <Window x:Class"ButtonClickSound.MainWindow"xmlns"http://schemas.microsoft.com/win…...

编写测试用例
测试用例(Test Case)是用于测试系统的要素集合 目录 编写测试用例作用 编写测试用例要包含七大元素 测试用例的设计方法 1、等价类法 2、边界值法 3、正交表法 4、判定表法 5、错误推测法 6、场景法 编写测试用例作用 1、确保功能全面覆盖…...
解释程序(Python)不需要生成机器码 逐行解析 逐行执行
在计算机组成原理中,解释程序(Interpreter)通常不会生成独立的机器码,但具体情况取决于解释器的实现方式。以下是详细分析: 1. 传统解释程序:不生成机器码 直接逐行执行: 经典的解释器ÿ…...

每日Prompt:隐形人
提示词 黑色棒球帽,白色抹胸、粉色低腰短裙、白色襪子,黑色鞋子,粉紅色背包,衣服悬浮在空中呈现动态姿势,虚幻引擎渲染风格,高清晰游戏CG质感,户外山林背景,画面聚焦在漂浮的衣服上…...

TensorFlow深度学习实战(19)——受限玻尔兹曼机
TensorFlow深度学习实战(19)——受限玻尔兹曼机 0. 前言1. 受限玻尔兹曼机1.1 受限玻尔兹曼机架构1.2 受限玻尔兹曼机的数学原理 2. 使用受限玻尔兹曼机重建图像3. 深度信念网络小结系列链接 0. 前言 受限玻尔兹曼机 (Restricted Boltzmann Machine, RB…...