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

【C++练级之路】【Lv.8】【STL】list类的模拟实现



快乐的流畅:个人主页


个人专栏:《C语言》《数据结构世界》《进击的C++》

远方有一堆篝火,在为久候之人燃烧!

文章目录

  • 引言
  • 一、结点
  • 二、迭代器
    • 2.1 成员变量与默认成员函数
    • 2.2 operator*
    • 2.3 operator->
    • 2.4 operator++
    • 2.5 operator- -
    • 2.6 relational operators
  • 三、list
    • 3.1 成员变量
    • 3.2 迭代器
      • 3.2.1 begin
      • 3.2.2 end
    • 3.3 默认成员函数
      • 3.3.1 constructor
      • 3.3.2 destructor
      • 3.3.3 copy constructor
      • 3.3.4 operator=
    • 3.4 修改
      • 3.4.1 insert
      • 3.4.2 push_front
      • 3.4.3 push_back
      • 3.4.4 erase
      • 3.4.5 pop_front
      • 3.4.6 pop_back
      • 3.4.7 clear
      • 3.4.8 swap
  • 总结

引言

因为list结构的特殊性,所以拆分为结点、迭代器和list本身进行学习。

一、结点

细节:

  1. 使用struct,标明公有属性(这样从外部调用比较方便
  2. list是带头双向循环链表
  3. 提供全缺省的默认构造函数
template<class T>
struct __list_node
{__list_node<T>* _prev;__list_node<T>* _next;T _data;__list_node(const T& x = T()): _prev(nullptr), _next(nullptr), _data(x){}
};

二、迭代器

由于list的每个结点物理空间不连续,导致迭代器不能像之前string、vector那样简单的设计为指针,而是设计为一个类(进行封装),以此完成*、->、++、–等一系列操作。

2.1 成员变量与默认成员函数

细节:

  1. 仍然使用struct,标明公有属性
  2. 成员变量是一个结点的指针
  3. 提供带参构造函数(其余的默认成员函数不用显式定义,浅拷贝即可)
template<class T, class Ref, class Ptr>
struct __list_iterator
{typedef __list_node<T> node;typedef __list_iterator<T, Ref, Ptr> self;node* _node;__list_iterator(node* n): _node(n){}
};

此时的迭代器设计,可以说是list乃至STL的精华,天才般地运用了类的优势。

2.2 operator*

细节:

  • 返回引用,为了区别普通迭代器和const迭代器
Ref operator*()
{return _node->_data;
}

2.3 operator->

细节:

  • 返回指针,为了区别普通迭代器和const迭代器
Ptr operator->()
{return &_node->_data;
}

2.4 operator++

细节:

  1. 为了区分前置和后置,后置参数加上int(无实际意义,以示区分)
  2. 前置传引用返回,后置传值返回
self& operator++()//前置++
{_node = _node->_next;return *this;
}self operator++(int)//后置++
{self tmp(*this);_node = _node->_next;return tmp;
}

2.5 operator- -

细节:同上

self& operator--()//前置--
{_node = _node->_prev;return *this;
}self operator--(int)//后置--
{self tmp(*this);_node = _node->_prev;return tmp;
}

2.6 relational operators

bool operator!=(const self& s)
{return _node != s._node;
}bool operator==(const self& s)
{return _node == s._node;
}

三、list

3.1 成员变量

list类包含了

  • _head(指向哨兵位)
template<class T>
class list
{
public:typedef __list_node<T> node;
private:node* _head;
};

3.2 迭代器

typedef __list_iterator<T, T&, T*> iterator;
typedef __list_iterator<T, const T&, const T*> const_iterator;

3.2.1 begin

细节:

  1. begin()在_head->next
  2. 使用匿名对象
iterator begin()
{return iterator(_head->_next);
}const_iterator begin() const
{return const_iterator(_head->_next);
}

3.2.2 end

细节:

  1. end()在_head
  2. 使用匿名对象
iterator end()
{return iterator(_head);
}const_iterator end() const
{return const_iterator(_head);
}

3.3 默认成员函数

3.3.1 constructor

空初始化:创建哨兵位

void empty_init()
{_head = new node;_head->_prev = _head;_head->_next = _head;
}

无参构造

list()
{empty_init();
}

迭代器区间构造

细节:使用类模板,可以传任意类型的迭代器

template <class InputIterator>
list(InputIterator first, InputIterator last)
{empty_init();while (first != last){push_back(*first);++first;}
}

3.3.2 destructor

~list()
{clear();delete _head;_head = nullptr;
}

3.3.3 copy constructor

现代写法

细节:

  1. 迭代器区间构造,构造出临时对象
  2. 再使用list中的swap,交换*this和tmp的值,完成拷贝构造
list(const list<T>& lt)
{empty_init();list<T> tmp(lt.begin(), lt.end());swap(tmp);
}

3.3.4 operator=

现代写法

细节:

  1. 传参变成传值,这样就会拷贝构造出一个临时对象
  2. 再使用list中的swap,交换*this和tmp的值,完成赋值重载
list<T>& operator=(list<T> lt)
{swap(lt);return *this;
}

3.4 修改

3.4.1 insert

指定位置插入

细节:

  1. 在pos之前插入
  2. 迭代器不会失效
void insert(iterator pos, const T& x)
{node* cur = pos._node;node* prev = cur->_prev;node* new_node = new node(x);prev->_next = new_node;new_node->_prev = prev;cur->_prev = new_node;new_node->_next = cur;
}

3.4.2 push_front

头插

void push_front(const T& x)
{insert(begin(), x);
}

3.4.3 push_back

尾插

void push_back(const T& x)
{insert(end(), x);
}

3.4.4 erase

指定位置删除

细节:

  1. assert断言,防止删除哨兵位
  2. 返回删除节点的下一位,防止迭代器失效
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;return iterator(next);
}

3.4.5 pop_front

void pop_front()
{erase(begin());
}

3.4.6 pop_back

void pop_back()
{erase(--end());
}

3.4.7 clear

清除所有结点(除哨兵位以外)

void clear()
{iterator it = begin();while (it != end()){erase(it++);}
}

3.4.8 swap

交换两个list类的值

细节:使用std库中的swap函数,交换各个成员变量的值

void swap(list<T>& lt)
{std::swap(_head, lt._head);
}

总结

学习完list类,对于STL中的精华——迭代器设计,有了更深一步的了解。同时,了解多参数模板的用途和方法,极大提高代码复用程度。


真诚点赞,手有余香

相关文章:

【C++练级之路】【Lv.8】【STL】list类的模拟实现

快乐的流畅&#xff1a;个人主页 个人专栏&#xff1a;《C语言》《数据结构世界》《进击的C》 远方有一堆篝火&#xff0c;在为久候之人燃烧&#xff01; 文章目录 引言一、结点二、迭代器2.1 成员变量与默认成员函数2.2 operator*2.3 operator->2.4 operator2.5 operator- …...

【右一的电子笔记】全导航,持续更新...

文章目录 &#x1f4da;计算机基础&#x1f407;高程&#xff08;c&#xff09;&#x1f407;python基础&#x1f407;数据结构&#x1f407;数据库系统概念&#x1f407;计算机网络&#x1f407;计算机组成原理&#x1f407;操作系统 &#x1f4da;大数据&#x1f407;大数据管…...

关于前端的console的方法的收集

console的常用方法列举 console.assert() 如果第一个参数为 false &#xff0c;则将消息和堆栈跟踪记录到控制台。 console.clear() 清空控制台&#xff0c;并输出 Console was cleared。 console.count() 以参数为标识记录调用的次数&#xff0c;调用时在控制台打印标识…...

大工程 从0到1 数据治理 数仓篇(sample database classicmodels _No.7)

大工程 从0到1 数据治理 之数仓篇 我这里还是sample database classicmodels为案列&#xff0c;可以下载&#xff0c;我看 网上还没有类似的 案列&#xff0c;那就 从 0-1开始吧&#xff01; 提示&#xff1a;写完文章后&#xff0c;目录可以自动生成&#xff0c;如何生成可参…...

phpcms v9敏感词内容替换

后台先在"扩展"——>"敏感词管理"中添加敏感词&#xff0c;然后修改phpcms\modules\content\content.php文件来实现添加或者编辑内容时敏感词的替换。&#xff08;如果涉及会员投稿和留言等&#xff0c;也需要在对应模块中做类似处理&#xff09; 在ad…...

浏览器---浏览器/http相关面试题

1.localStorage和sessionStorage 共同点&#xff1a;二者都是以key-value的键值对方式存储在浏览器端&#xff0c;大小大概在5M。 区别&#xff1a; &#xff08;1&#xff09;数据有效期不同&#xff1a;sessionStorage仅在当前浏览器窗口关闭之前有效&#xff1b;localStorag…...

java 中开源的html解析库Jsoup 简单例子

下面是一个使用Jsoup库解析HTML的简单Java例子。这个例子展示了如何使用Jsoup从一个HTML字符串中提取数据。 首先&#xff0c;确保你已经将Jsoup作为依赖项添加到你的项目中。如果你使用的是Maven&#xff0c;可以在pom.xml文件中添加以下依赖&#xff1a; &…...

Java程序中为什么要使用StringBuilder

遇到这个问题是来源于leetcode的一道题&#xff1a;字符串解码。其中的题解涉及字符串的操作使用的是StringBuilder&#xff0c;不是String。 class Solution {public String decodeString(String s) {StringBuilder res new StringBuilder();int multi 0;LinkedList<Int…...

【软件架构】02-复杂度来源

1、性能 1&#xff09;单机 受限于主机的CPU、网络、磁盘读写速度等影响 在多线程的互斥性、并发中的同步数据状态等&#xff1b; 扩展&#xff1a;硬件资源、增大线程池 2&#xff09;集群 微服务化拆分&#xff0c;导致调用链过长&#xff0c;网络传输的消耗过多。 集…...

怎样让MCU/SFU视频会议ovmedia 接入GB28281监控视频参会互动

在国内视频应用对GB监控接入是常规操作&#xff0c;很多系统需要接入监控视频交互处理。我们以ovmedia视频会议为例做一个接入互动。 GB28181协议在流媒体系统较为普及&#xff0c;我们以开源SRS系统对接监控端再接入会议&#xff08;也可以用商用GB流平台&#xff0c;操作基本…...

Spring Boot打war包部署到Tomcat,访问页面404 !!!

水善利万物而不争&#xff0c;处众人之所恶&#xff0c;故几于道&#x1f4a6; 文章目录 Spring Boot打war包部署到Tomcat&#xff0c;访问页面404 &#xff01;&#xff01;&#xff01;解决办法&#xff1a;检查Tomcat版本和Jdk的对应关系&#xff0c;我的Tomcat是6.x&#x…...

Docker Desktop 4.27.1 Windows 10 安装 教程

Docker Desktop 4.27.1 Windows 10 安装 版本要求windows 版本要求wsl 版本要求docker desktop 版本 安装首先确保系统版本符合要求前提下安装wsl安装 Dockers Desktop安装说明 安装问题docker Desktop 无法正常启动&#xff0c;提示wsl 相关信息wsl --install 执行输出帮助日志…...

【ARMv8M Cortex-M33 系列 8 -- RT-Thread 移植 posix pthread】

文章目录 RT-Thread POSIX PthreadRT-Thread Pthread 相关宏定义RT-Thread libc 初始化RT-Thread Pthread 测试 RT-Thread POSIX Pthread pthread是POSIX&#xff08;Portable Operating System Interface&#xff09;标准定义的一套线程相关的API&#xff0c;全称为POSIX Thr…...

fastApi笔记08-Cookie和Header

Cookie 可以像Query&#xff0c;Path&#xff0c;Body等同样的方式来定义Cookie参数 from typing import Annotatedfrom fastapi import Cookie, FastAPIapp FastAPI()app.get("/items/") async def read_items(ads_id: Annotated[str | None, Cookie()] None):r…...

解决pycharm中PIL安装失败

问题&#xff1a;在调用pil时显示pil标红 我在设置中下载每次失败&#xff0c;显示 ERROR: Could not find a version that satisfies the requirement PIL (from versions: none) ERROR: No matching distribution found for PIL我尝试了很久&#xff0c;查看了一些博客 &a…...

数据结构哈希表

这里个大家用数组来模拟哈希表 法一&#xff1a;拉链法 法二&#xff1a;开放寻址法 /** Project: 11_哈希表* File Created:Sunday, January 17th 2021, 2:11:23 pm* Author: Bug-Free* Problem:AcWing 840. 模拟散列表 拉链法*/ #include <cstring> #include <iostr…...

[C#]winform使用引导APSF和梯度自适应卷积增强夜间雾图像的可见性算法实现夜间雾霾图像的可见度增强

【算法介绍】 提升夜间雾霾图像可见度的技术研究&#xff1a;引导APSF与梯度自适应卷积的应用 随着城市化的快速发展&#xff0c;雾霾现象日益严重&#xff0c;尤其是在夜间&#xff0c;雾霾对图像的可见度造成了极大的影响。因此&#xff0c;提升夜间雾霾图像的可见度成为了…...

【Django】Django自定义后台表单——对一个关联外键对象同时添加多个内容

以官方文档为例&#xff1a; 一个投票问题包含多个选项&#xff0c;基本的表单设计只能一个选项一个选项添加&#xff0c;效率较低&#xff0c;如何在表单设计中一次性添加多个关联选项&#xff1f; 示例代码&#xff1a; from django.contrib import adminfrom .models impo…...

迷茫?没有努力的方向?没有耐心去坚持?精选书籍推荐2

迷茫书籍推荐 在渡过自卑期后&#xff0c;下一阶段就是迷茫期&#xff0c;我就是典型。坚持考研失败&#xff0c;然后工作上不顺利&#xff0c;尽管稍稍改变了自卑&#xff0c;但是却因为从前的失败&#xff0c;对下一步何去何从产生了迷茫。这也是我这篇文章希望帮助大家解决的…...

MySQL报错:sql_mode=only_full_group_by解决方法

Linux环境 ubuntu 22.04 MySQL是8.0.35版本 问题描述 Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column auth_system.t_class_temp_config.id which is not functionally dependent on columns in GROUP BY clause; this is inco…...

内存暴涨却查无踪迹?Python对象生命周期管理的7个致命盲区,现在不看明天宕机!

第一章&#xff1a;Python智能体内存管理的核心原理Python智能体&#xff08;如基于LangChain、LlamaIndex构建的Agent&#xff09;在运行过程中并非仅依赖语言模型推理&#xff0c;其内存管理机制直接决定状态持久性、上下文感知能力与多轮交互一致性。核心在于Python对象生命…...

Qwen3-Embedding-4B广告过滤应用:恶意内容识别系统实战

Qwen3-Embedding-4B广告过滤应用&#xff1a;恶意内容识别系统实战 1. 引言&#xff1a;当广告变成“牛皮癣”&#xff0c;我们如何反击&#xff1f; 想象一下&#xff0c;你运营着一个用户社区或内容平台。每天&#xff0c;用户都在热情地分享、讨论。但总有一些不速之客&am…...

STM32CubeMX配置EXTI中断,别再在HAL_GPIO_EXTI_Callback里用HAL_Delay了!

STM32外部中断实战&#xff1a;避开HAL_Delay陷阱的三种解决方案 第一次在STM32项目中使用外部中断时&#xff0c;我遇到了一个令人困惑的问题——按下按键后程序突然卡死。经过反复排查&#xff0c;最终发现问题出在中断回调函数中的HAL_Delay调用上。这个看似简单的延时函数&…...

06_gstack发布运营:一键发布与文档同步机制

06_gstack发布运营&#xff1a;一键发布与文档同步机制关键字&#xff1a;gstack、一键发布、ship技能、document-release、文档同步、发布流水线、CHANGELOG、PR自动化、retro、工程回顾你上一次修改完代码到实际提交 PR&#xff0c;中间经历了多少步&#xff1f; git stash&a…...

从nvidia-smi到npu-smi:给CUDA开发者的华为昇腾NPU监控指南

从nvidia-smi到npu-smi&#xff1a;CUDA开发者快速掌握昇腾NPU监控的实战手册 当你的技术栈从英伟达GPU扩展到华为昇腾NPU时&#xff0c;监控工具的使用体验就像从自动挡切换到手动挡——虽然最终目的地相同&#xff0c;但操作逻辑需要重新适应。作为曾经每天与nvidia-smi打交道…...

Tauri开发手记——1.从零到一:环境搭建与首次构建实战

1. 环境准备&#xff1a;从零搭建Tauri开发环境 第一次接触Tauri开发时&#xff0c;环境搭建往往是最让人头疼的环节。作为一个跨平台桌面应用框架&#xff0c;Tauri需要同时处理前端和后端&#xff08;Rust&#xff09;的依赖关系。我在Windows系统上踩过不少坑&#xff0c;现…...

模型调参实战指南:Temperature、Top-k与Top-p的黄金组合法则

1. 理解三大核心参数&#xff1a;从理论到实践 第一次接触大模型调参时&#xff0c;我被Temperature、Top-k和Top-p这三个参数搞得晕头转向。直到在真实项目中踩过几次坑后才明白&#xff0c;它们就像烹饪中的"盐、糖、醋"——看似简单&#xff0c;但配比不同就能产生…...

Mastering nohup: Redirecting Output for Persistent Server Deployments

1. 为什么你需要掌握nohup命令 想象一下这个场景&#xff1a;你在远程服务器上启动了一个重要的Java服务&#xff0c;花了半小时调试终于跑起来了。这时候老板喊你开会&#xff0c;你顺手关闭了终端窗口。等会议结束回来一看——服务居然挂了&#xff01;所有努力付诸东流&…...

基于模型参考的滑模控制/MRSMC 基于模型参考的滑模控制(MRSMC, Model Refe...

基于模型参考的滑模控制/MRSMC 基于模型参考的滑模控制&#xff08;MRSMC, Model Reference Sliding Mode Control&#xff09;是一种结合了模型参考控制和滑模控制优点的控制策略。 它通常用于系统的鲁棒控制&#xff0c;尤其是在面对模型不确定性和外部扰动时。 在simulink中…...

30%重复率的论文如何快速合格?爱毕业aibye的AI改写工具提供五条建议

嘿&#xff0c;大家好&#xff01;我是AI菌。今天咱们来聊聊一个让无数学生头疼的问题&#xff1a;论文重复率飙到30%以上怎么办&#xff1f;别慌&#xff0c;我这就分享5个实用降重技巧&#xff0c;帮你一次搞定&#xff0c;轻松压到合格线以下。这些方法都是我亲身试验过的&a…...