C++list模拟实现
C++list模拟实现
- list接口总结
- 结点类的模拟实现
- 迭代器的模拟实现
- 迭代器模板参数
- 迭代器类中的构造函数
- 迭代器类中的运算符重载
- operator++和operator - -
- operator!= 和operator==
- operator*
- operator->
- 总览
- list 类
- 构造函数
- 拷贝构造函数
- 赋值运算符重载operator=
- clear()
- 析构函数
- 迭代器相关函数
- begin和end
- 与容器修改有关函数
- insert 插入
- erase 删除
- push_back()尾插
- pop_back 尾删
- push_front 头插
- pop_front 头删
- size
list接口总结
namespace bite
{// List的节点类template<class T>struct ListNode{ListNode(const T& val = T());ListNode<T>* _pre;ListNode<T>* _next;T _date;};//List的迭代器类template<class T, class Ref, class Ptr>class Listiterator{typedef ListNode<T>* Node;typedef Listiterator<T, Ref, Ptr> Self;public:Listiterator(Node* Node);Ref operator*();Ptr operator->();Self& operator++();dSelf operator++(int);Self& operator--();Self& operator--(int);bool operator!=(const Self& l);bool operator==(const Self& l);private:Node _node;};//list类template<class T>class list{typedef ListNode<T> Node;public:typedef Listiterator<T, T&, T*> iterator;typedef Listiterator<T, const T&, const T&> const_iterator;public:///// List的构造void init();list();list(const list<T>& val)list<T>& operator==(list<T> val)~list();///// List Iteratoriterator begin();iterator end();const_iterator begin();const_iterator end();///// List Capacitysize_t size()const;bool empty()const;// List Modifyvoid push_back(const T& val) { insert(end(), val); }void pop_back() { erase(--end()); }void push_front(const T& val) { insert(begin(), val); }void pop_front() { erase(begin()); }// 在pos位置前插入值为val的节点iterator insert(iterator pos, const T& val);// 删除pos位置的节点,返回该节点的下一个位置iterator erase(iterator pos);void clear();void swap(list<T>& lt);private:Node* _head;size_t _size;};
};
结点类的模拟实现
STL中的list为带头双向循环链表。
如图:
所以我们要实现list链表之前还要实现结点类。 结点中要存储前一个结点的地址和后一个结点的地址、还有数据。因此我们只需要设定成员变量和构造函数就行。
template <class T> //设定模板结点中的数据类型可以是不同的
struct ListNode
{typedef ListNode<T> Node;ListNode(const T& val = T())//构造函数:_prev(nullptr),_next(nullptr),_date(val){}Node* _prev;//前驱指针Node* _next;//后驱指针T _date;//数据};
迭代器的模拟实现
我们在string和vector中迭代器模拟都是直接给定模板指针。但是到了list中我们就需要实现一个迭代器类型。why?
这是因为我们在string和vector中数据的存储都是存放在一块连续的空间。我们可以同过指针进行对数据的增删查改。
list并不是存放一块连续的空间,而是通过结点存放下一个结点的地址一个一个连结起来的。 所以我们不能通过指针来实现链表的增删查改。
迭代器是一种用于遍历集合或容器中元素的接口,它可以隔离对容器的访问方式和底层实现,从而实现解耦。迭代器可以依次访问容器中的每一个元素,而无需了解容器的内部细节。
既然我们的指针不能满足迭代器的行为,那我们就将指针封装成类,使其能够满足迭代器的需要。
迭代器模板参数
迭代器我们给处三个模板参数。
template<class T, class Ref, class Ptr>
实现两个迭代器,一个iterator 和另一个 const_iterator。
typedef Listiterator<T, T&, T*> iterator;typedef Listiterator<T, const T&, const T&> const_iterator;
从这里我们可以看出,Ref 代表的是引用类型和指针类型。
迭代器类中的构造函数
我们迭代器的底层就是对指针进行的封装,因此成员变量就一个就是结点指针。
Listiterator(Node* node):_node(node)
{}
迭代器类中的运算符重载
operator++和operator - -
list的迭代器没必要实现operator+,只需要实现自增和和自减
自增:
//前置++
self& operator++()
{_node = _node->_next;return *this;
}
//后置++
self operator++(int)
{self tmp = *this;_node = _node->_next;return tmp;
}
自减:
//前置--
self& operator--()
{_node = _node->_prev;return *this;
}
//后置--
self operator--(int)
{self tmp = *this;_node = _node->_prev;return tmp;
}
operator!= 和operator==
实现两个迭代器之间的比较。
bool operator!=(const self& val)
{return _node != val._node;
}bool operator==(const self& val)
{return _node == val._node;
}
operator*
当我们使用解引用操作符,想要得到的是这个地址所指向的数据。所以我们operator* 返回该地址所指向的数据就行。
Ref operator*()
{return _node->_date;
}
operator->
我们在使用list时,如果list内结点存储的数据不是内置类型,而是自定义类型,例如Date日期类。我们访问Date类中的成员,就需要用到operator->。
例如:
struct A //自定义类型
{A(int x=0 ,int y =0):_aa1(x),_aa2(y){}int _aa1;int _aa2;
};
void test()
{A aa1(0,0);list<A> lt;lt.push_back(aa1);lt.push_back(A(1, 1));lt.push_back({ 2,2 });lt.push_back({3,3});list<A>::iterator it = lt.begin();while (it != lt.end()){cout << it->_aa1 << " " << it->_aa2 << endl;;it++;}
}
operator->只需要返回结点数据的地址就行。
Ptr operator->()
{return &_node->_date;
}
这里有人就可能发现,这种方式访问自定义类型成员需要用到两个->。
这里其实是忽略了一个->,两个->第一个运算符重载的调用是指向A* ,另一个是原生指针的调用。为了让代码有良好的可读性,编译器做了特殊处理,忽略了一个->。
总览
template <class T , class Ref , class Ptr>
struct Listiterator
{typedef ListNode<T> Node;typedef Listiterator<T , Ref , Ptr> self;Node* _node;Listiterator(Node* node):_node(node){}Ref operator*(){return _node->_date;}Ptr operator->(){return &_node->_date;}self& operator++(){_node = _node->_next;return *this;}self operator++(int){self tmp = *this;_node = _node->_next;return tmp;}self& operator--(){_node = _node->_prev;return *this;}self operator--(int){self tmp = *this;_node = _node->_prev;return tmp;}bool operator!=(const self& val){return _node != val._node;}bool operator==(const self& val){return _node == val._node;}};
list 类
构造函数
list是一个双向带头循环链表,list初始化需要设置一个哨兵位,并使其的前驱指针和后驱指针都指向自己。
void init()
{_head = new Node;_head->_prev = _head;_head->_next = _head;_size = 0;
}
list()
{init();
}
拷贝构造函数
list的拷贝构造就是根据所给的对象,拷贝处一个新的对象。拷贝构造函数我们先给一个头结点使其前驱指针和后驱指针都指向自己,然后再遍历对象,让对象中的结点一个一个尾插入到新的list中去。
list(const list<T>& val)
{init();for (auto& e : val){push_back(e);}
}
赋值运算符重载operator=
我们这里直接使用现代写法。利用传入的参数通过编译器自动调用list构造函数构造出一个list对象,
再使用swap使其与原来的容器进行交换。
list<T>& operator==(list<T> val){swap(val);return *this;}
clear()
clear函数只需要将链表中的结点和数据清理即可,给一个迭代器使其指向头结点,然后遍历链表将结点一个一个删除,除了哨兵位以外。
void clear()
{iterator it = begin();while (it != end()){it = erase(it);}
}
析构函数
析构函数首先我们先将list容器中的数据清除,让后在释放哨兵位和让_size归零。
~list()
{clear();delete _head;_head = nullptr;_size = 0;
}
迭代器相关函数
begin和end
在这里我们需要清楚的是begin为哨兵位的下一个结点,而end就是哨兵位。
iterator begin()
{return _head->_next;
}iterator end()
{return _head;
}
还有const begin 和end
const_iterator begin() const
{return _head->_next;
}const_iterator end() const
{return _head;
}
与容器修改有关函数
insert 插入
insert函数作用是在pos位置插入一个新的结点。
如图:
我们要在pos位置插入一个新的结点(newnode)就得让新结点前驱指针指向pos结点前一个结点(prev),让prev的后驱指针指向newnode。newnode后驱指针指向pos结点,pos前驱指针指向newnode。
void insert(iterator pos , const T& val)
{Node* newnode = new Node(val);Node* cur = pos._node;Node* prev = cur->_prev;newnode->_next = cur;cur->_prev = newnode;newnode->_prev = prev;prev->_next = newnode;_size++;}
erase 删除
erase就是删除pos位置的结点。
如图:
iterator erase(iterator pos)
{assert(pos != _head);Node* del = pos._node;Node* prev = del->_prev;Node* next = del->_next;prev->_next = next;next->_prev = prev;delete del;--_size;return iterator(next);
}
注意: erase还要注意迭代器失效的问题因此我们的返回值指向的是下一个结点。
push_back()尾插
双向带头循环链表的尾插比较容易理解。新建一个结点使其的前驱指针指向尾结点,尾结点的后驱结点指向新的结点。新结点的后驱指针指向哨兵位,哨兵位的前驱指针指向新结点,再++_size。
void push_back(const T& val)
{Node* newnode = new Node(val);Node* tail = _head->_prev;newnode->_prev = tail;tail->_next = newnode;newnode->_next = _head;_head->_prev = newnode;_size++;
}
还可以复用insert函数,是push_back函数变得简单。
void push_back(const T& val) { insert(end(), val); }
pop_back 尾删
我们这里直接复用erase函数。
void pop_back(){erase(--end()); }
push_front 头插
头插也是复用insert函数。
void push_front(const T& val)
{ insert(begin(), val);
}
pop_front 头删
void pop_front(){ erase(begin()); }
size
返回结点个数。
size_t size()
{return _size;
}
相关文章:
C++list模拟实现
Clist模拟实现 list接口总结结点类的模拟实现迭代器的模拟实现迭代器模板参数迭代器类中的构造函数迭代器类中的运算符重载operator和operator - -operator! 和operatoroperator*operator->总览 list 类构造函数拷贝构造函数赋值运算符重载operatorclear…...
设计模式(22):解释器模式
解释器 是一种不常用的设计模式用于描述如何构成一个简单的语言解释器,主要用于使用面向对象语言开发的解释器和解释器设计当我们需要开发一种新的语言时,可以考虑使用解释器模式尽量不要使用解释器模式,后期维护会有很大麻烦。在项目中&…...
kubernetes docker版本安装测试
文章目录 测试环境kubernetes安装环境配置安装程序下载镜像初始化reset环境init构建kubernetes配置授权信息配置网络插件查看状态 简单实例测试 测试环境 [rootlocalhost ~]# cat /etc/redhat-release CentOS Linux release 7.9.2009 (Core)kubernetes安装 参考kuberneter文档…...
策略模式:灵活调整算法的设计精髓
在软件开发中,策略模式是一种行为型设计模式,它允许在运行时选择算法的行为。通过定义一系列算法,并将每个算法封装起来,策略模式使得算法可以互换使用,这使得算法可以独立于使用它们的客户。本文将详细介绍策略模式的…...
[INS-30014]无法检查指定的位置是否位于 CFS 上
文章目录 一、具体错误二、通用解决方案1、可能的问题原因2、解决方案3、常见原因之hosts文件配置问题hosts配置方法hosts文件不可编辑解决办法 一、具体错误 在安装ORACLE19c的时候,出现无法检查指定的位置是否位于CFS上 二、通用解决方案 1、可能的问题原因 遇…...
机器学习和深度学习 -- 李宏毅(笔记与个人理解)Day 13
Day13 Error surface is rugged…… Tips for training :Adaptive Learning Rate critical point is not the difficult Root mean Square --used in Adagrad 这里为啥是前面的g的和而不是直接只除以当前呢? 这种方法的目的是防止学习率在训练过程中快速衰减。如果只用当前的…...
[Python图像识别] 五十二.水书图像识别 (2)基于机器学习的濒危水书古文字识别研究
该系列文章是讲解Python OpenCV图像处理知识,前期主要讲解图像入门、OpenCV基础用法,中期讲解图像处理的各种算法,包括图像锐化算子、图像增强技术、图像分割等,后期结合深度学习研究图像识别、图像分类应用。目前我进入第二阶段Python图像识别,该部分主要以目标检测、图像…...
Jmeter针对多种响应断言的判断
有时候response返回的结果并非一种,有多种,需要对这几种进行判断的时候需要使用Bean Shell。 (1)首先获取响应数据 String response prev.getResponseDataAsString(); ResponseCode 响应状态码 responseHeaders 响应头信息 res…...
Harmony鸿蒙南向驱动开发-Regulator接口使用
功能简介 Regulator模块用于控制系统中某些设备的电压/电流供应。在嵌入式系统(尤其是手机)中,控制耗电量很重要,直接影响到电池的续航时间。所以,如果系统中某一个模块暂时不需要使用,就可以通过Regulato…...
【opencv】示例-grabcut.cpp 使用OpenCV库的GrabCut算法进行图像分割
left mouse button - set rectangle SHIFTleft mouse button - set GC_FGD pixels CTRLleft mouse button - set GC_BGD pixels 这段代码是一个使用OpenCV库的GrabCut算法进行图像分割的C程序。它允许用户通过交互式方式选择图像中的一个区域,并利用GrabCut算法尝试…...
GEE数据集——巴基斯坦国家级土壤侵蚀数据集(2005 年和 2015 年)
简介 巴基斯坦国家级土壤侵蚀数据集(2005 年和 2015 年) 该数据集采用修订的通用土壤流失方程 (RUSLE),并考虑了六个关键影响因素:降雨侵蚀率 (R)、土壤可侵蚀性 (K)、坡长 (L)、坡陡 (S)、覆盖管理 (C) 和保护措施 (P)ÿ…...
服务器代理
服务器代理 配置:64G内存1 3090(24g)1P4000(8g) SSH连接 工作路径:/home/ubuntu/workspace/python Anaconda路径:/home/Ubuntu 1.在工作路径下创建自己的文件夹作为workspace 2.以用户ubunbtu登…...
【SGDR】《SGDR:Stochastic Gradient Descent with Warm Restarts》
arXiv-2016 code: https://github.com/loshchil/SGDR/blob/master/SGDR_WRNs.py 文章目录 1 Background and Motivation2 Related Work3 Advantages / Contributions4 Method5 Experiments5.1 Datasets and Metric5.2 Single-Model Results5.3 Ensemble Results5.4 Experiment…...
如何将arping以及所有依赖打包安装到另外一台离线ubuntu机器
ubuntu系统下可以使用arping命令检测局域网内一些ip是否冲突,使用方式为: arping xx.xx.xx.xx 在线情况下,可以使用下面命令下载arping,然后使用即可 apt install arping 但是有些情况下机器可能不能上网,这时就需要将…...
mac上如何安装python3
mac上如何安装python3? 安装homebrew 在终端执行命令 /usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)" 执行完成后,homebrew和pip等工具就自动安装好了。 接下来安装python3.在终端…...
Java 那些诗一般的 数据类型 (下篇)
本篇会加入个人的所谓鱼式疯言 ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. 🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人能接…...
WEB3.0:互联网的下一阶段
随着互联网的发展,WEB3.0时代正在逐步到来。本文将深入探讨WEB3.0的定义、特点、技术应用以及未来展望,为读者带来全新的思考。 一、什么是WEB3.0? WEB3.0可以被理解为互联网发展的下一阶段,是当前WEB2.0的升级版。相较于2.0时代…...
Fastgpt配合chatglm+m3e或ollama+m3e搭建个人知识库
概述: 人工智能大语言模型是近年来人工智能领域的一项重要技术,它的出现标志着自然语言处理领域的重大突破。这些模型利用深度学习和大规模数据训练,能够理解和生成人类语言,为各种应用场景提供了强大的文本处理能力。AI大语言模…...
如何使用选择器精确地控制网页中每一个元素的样式?
1. 基础知识 什么是 CSS 元素选择器 CSS 元素选择器是一种在网页中通过元素类型来应用样式的方法。 简单来说,它就像是一个指挥棒,告诉浏览器哪些 HTML 元素需要应用我们定义的 CSS 样式规则。 为何要使用 CSS 元素选择器 使用元素选择器可以让我们…...
各个微前端框架的优劣浅谈
各个微前端框架都有其独特的优势和劣势,下面我将针对几个主流的微前端框架进行简要的优劣分析: single-spa 优势: 轻量级:single-spa是一个非常轻量级的微前端框架,它主要提供了一个加载和管理微应用的机制,…...
自动化运维(二十二)Ansible实战 之Jenkins模块
Ansible提供了一些模块,可以用来与Jenkins进行交互,执行各种操作,如创建任务、触发构建、获取构建结果等。通过使用这些模块,我们可以将Jenkins的配置和管理集成到Ansible的自动化流程中。 以下是一些常用的Ansible Jenkins模块: 1、jenkins_job模块 jenkins_job模块用于创建…...
Python数据分析与应用 |第4章 使用pandas进行数据预处理 (实训)
表1-1healthcare-dataset-stroke.xlsx 部分中风患者的基础信息和体检数据 编号性别高血压是否结婚工作类型居住类型体重指数吸烟史中风9046男否是私人城市36.6以前吸烟是51676女否是私营企业农村N/A从不吸烟是31112男否是私人农村32.5从不吸烟...
基于双向长短期神经网络BILSTM的线损率预测,基于gru的线损率预测
目录 背影 摘要 LSTM的基本定义 LSTM实现的步骤 BILSTM神经网络 基于双向长短期神经网络BILSTM的线损率预测,基于gru的线损率预测 完整代码:基于双向长短期神经网络BILSTM的线损率预测,基于gru的线损率预测(代码完整,数据齐全)资源-CSDN文库 https://download.csdn.net/d…...
智能售货机:引领便捷生活
智能售货机:引领便捷生活 在这个科技迅速进步的时代,便捷已成为生活的必需。智能售货机作为技术与便利完美结合的产物,正逐渐改变我们的购物方式,为都市生活增添新的活力。 智能售货机的主要优势是它的极致便利性。不论是在地铁…...
正向代理和反向代理
正向代理和反向代理是网络中常见的两种代理方式,它们在网络通信中扮演着不同的角色。 正向代理: 正向代理是代理服务器位于客户端和目标服务器之间的一种代理方式。 客户端向代理服务器发送请求,然后代理服务器将请求转发给目标服务器&…...
kimichat使用技巧:用语音对话聊天
kimichat之前是只能用文字聊天的,不过最近推出了语音新功能,也可以用语音畅快的对话聊天了。 这个功能目前支持手机app版本,所以首先要在手机上下载安装kimi智能助手。已经安装的,要点击检查更新,更新到最新的版本。 …...
机器学习-09-图像处理02-PIL+numpy+OpenCV实践
总结 本系列是机器学习课程的系列课程,主要介绍机器学习中图像处理技术。 参考 【人工智能】PythonOpenCV图像处理(一篇全) 一文讲解方向梯度直方图(hog) 【杂谈】计算机视觉在人脸图像领域的十几个大的应用方向&…...
应急响应-战前反制主机HIDSElkeid蜜罐系统HFish
知识点 战前-反制-平台部署其他更多项目: https://github.com/birdhan/SecurityProduct HIDS:主机入侵检测系统,通常会有一个服务器承担服务端角色,其他主机就是客户端角色,客户端加入到服务端的检测范围里ÿ…...
C#:24小时制和12小时制之间的转换
任务描述 本关任务:编写一个程序,利用求余运算完成24小时制和12小时制之间的转换。 注意:要求输入的数字是0到24之间的整数。 测试说明 平台会对你编写的代码进行测试: 测试输入:4 预期输出: 现在是上午4…...
说说TCP为什么需要三次握手和四次挥手?
文章目录 一、三次握手为什么不是两次握手? 二、四次挥手四次挥手原因 三、总结参考文献 一、三次握手 三次握手(Three-way Handshake)其实就是指建立一个TCP连接时,需要客户端和服务器总共发送3个包 主要作用就是为了确认双方的接收能力和…...
互联网装修服务平台/新人学会seo
本文为 AI 研习社编译的技术博客,原标题 :Python: How To Reduce Memory Consumption By Half By Adding Just One Line Of Code?作者 | Alex Maison翻译 | 邓普斯•杰弗校对 | 酱番梨 整理 | 菠萝妹原文链接:https://medium.com/alexmaisiu…...
网站开发需会的课程/青岛网站建设公司电话
4.1 开发完第一个鸿蒙应用后,下面在了解一下完整的鸿蒙应用打包发布后应该是什么样子:一个完整的打包后应用结构如下图所示,这里我们先了解结构,具体怎么打包很简单只要前提是要签名!1. HAP的分类HAP又可分为entry和feature两种模…...
做网站的 简历/临沂做网络优化的公司
霍夫变换将笛卡尔坐标系的直线用统计展示坐标系A中的点坐标系B中的线坐标系A中的线坐标系B中的点A中多点的连线B中多曲线的交点先理解这样一个思维左边的x-y坐标系中,设定k,b定值,则有一系列x,y值组成直线。相应的在右边k-b坐标系中ÿ…...
都匀住房和城乡建设部网站/搜索引擎营销是指
android默认的视频采集格式是NV21,(属于YUV420) 在onPreviewFrame中传进来的byte[] data即为NV21格式。 旋转算法 对NV21进行顺时针旋转90度,180度和270度算法。 旋转90度 privatebyte[] rotateYUV420Degree90(byte[] data, int i…...
网页设计作业样例/seo基础教程
光敏电阻器(photoresistor)又叫光感电阻,它的电阻值会根据光的强度变化而变化。 现在我要把它接入树莓派Pico,观察电阻值随光强度的变化情况,本试验参考的文章:https://peppe8o.com/how-to-use-a-photoresi…...
中国建设人才服务信息网是不是假冒网站/网络营销与管理
什么是Vue.nextTick() 官方文档解释如下: 在下次 DOM 更新循环结束之后执行延迟回调。在修改数据之后立即使用这个方法,获取更新后的 DOM。 我理解的官方文档的这句话的侧重点在最后那半句获取更新后的DOM,获取更新后的DOM言外之意就是什么操…...