通过比较list与vector在简单模拟实现时的不同进一步理解STL的底层

 cplusplus.com/reference/list/list/?kw=list

当我们大致阅读完list的cplusplus网站的文档时,我们会发现它提供的接口大致上与我们的vector相同。当然的,在常用接口的简单实现上它们也大体相同,但是它们的构造函数与迭代器的实现却大有不同。(食用本文时建议与文末的模拟实现代码一起食用,效果更佳)

一,list与vector在构造函数上的不同

 1.1成员封装的不同

我们在vector中,需要封装的成员只有我们顺序表的起始指针,有效元素的末尾指针,以及用来记录空间结束位置的指针:

如果我们在list中,便无法再这样封装,因为我们的list存放数据的内存并不连续,所以我们需要对链表的节点用额外的结构体封装,而在原本的list类里面我们封装的成员则是链表的头结点:

template<class T>
struct list_node
{T _data;list_node<T>* _next;list_node<T>* _prev;};
template<class T>
class list
{
void empty_init()
{_head = new Node();_head->_next = _head;_head->_prev = _head;
}
list()
{empty_init();
}
typedef list_node<T> Node;
private:Node* _head;
};

1.2迭代器的封装的不同 

在vector中,迭代器可以直接简单的设置为我们存储数据类型的对应指针。但是在list中,我们发现迭代器需要指向的是一个节点,有人说那我直接照搬vector不也OK。但是这时候就会有个问题,因为我们的节点相当于一个自定义类型,后面在进行调用的时候不可避免的会去调用它的构造函数,同时我们迭代器又会有许多的函数去使用,所以迭代器也需要额外的用一个类来封装。

template<class T, class Ref, class Ptr>
struct list_iterator//tip
{typedef list_node<T> Node;typedef list_iterator<T, Ref, Ptr> Self;
};

TIPS:这里将迭代器与结点成员设置为公开是为了方便list类的调用,实际上我们在使用list时也无法直接调用这些成员,这也是对成员的保护方式。

二,list与vector在迭代器上的不同(重点) 

 2.1对迭代器的const修饰

在我们的vector中,由于我们的迭代器本身就是我们存储数据的类型的相应指针,所以我们可以通过直接加const的方式来实现我们的const_iterator。但是在list中,由于我们的迭代器是指向一个自定义类型的指针,而我们的自定义类型中存储数据。如果我们直接用const来修饰,会发现我们此时无法修改迭代器指向的结点,从而无法完成我们后续的遍历。如果我们要为const来再额外封装一个类,会使代码看上去非常冗余。

在stl的源码中有着这样的几行代码:

template <class T, class Ref, class Ptr>
struct __slist_iterator : public __slist_iterator_base
{typedef __slist_iterator<T, T&, T*>             iterator;typedef __slist_iterator<T, const T&, const T*> const_iterator;typedef __slist_iterator<T, Ref, Ptr>           self;typedef T value_type;typedef Ptr pointer;typedef Ref reference;typedef __slist_node<T> list_node;

我们发现,它的模板参数有三个。其实由于我们的目的是为了源结点中的值无法被改变,只需要我们在返回结点中的值时加上const修饰,而我们获取存储数据的方式无非两种,一种是对迭代器解引用(其中_node是当前迭代器指向的结点):

T& operator*()
{return _node->_data;
}

或者通过->来获取:

T* operator->()
{return &_node->_data;
}

 所以我们只需要对T*与T&在模板实例化时用const修饰即可:

typedef list_iterator<T,T&,T*> iterator;//tip
typedef list_iterator<T,const T&,const T*> const_iterator;//tip

 TIP:在迭代器的类型中我们又分为随机,双向和单向迭代器,从左向右为父级关系。在使用库中的sort时对list无法使用快排,因为他是双向迭代器,而vector之所以可以使用是因为他是随机迭代器。

2.2list反向迭代器的实现

通过上面的多个模板参数的引出,我们可以对反向的迭代器Reverse_iterator来封装进行封装:

template<class Iterator>
class ReverseListIterator
{// 注意:此处typename的作用是明确告诉编译器,Ref是Iterator类中的类型,而不是静态成员变量// 否则编译器编译时就不知道Ref是Iterator中的类型还是静态成员变量// 因为静态成员变量也是按照 类名::静态成员变量名 的方式访问的
public:typedef typename Iterator::Ref Ref;typedef typename Iterator::Ptr Ptr;typedef ReverseListIterator<Iterator> Self;
public://// 构造ReverseListIterator(Iterator it) : _it(it) {}//// 具有指针类似行为Ref operator*() {Iterator temp(_it);--temp;return *temp;}Ptr operator->() { return &(operator*()); }//// 迭代器支持移动Self& operator++() {--_it;return *this;}Self operator++(int) {Self temp(*this);--_it;return temp;}Self& operator--() {++_it;return *this;}Self operator--(int){Self temp(*this);++_it;return temp;}//
// 迭代器支持比较
bool operator!=(const Self& l)const{ return _it != l._it;}
bool operator==(const Self& l)const{ return _it != l._it;}
Iterator _it;
};

原理其实就是我们2.1中介绍到的,这里我们直接给出模拟实现代码。

三,list与vector其他方面不同的总结(不仅是模拟实现上)

附件:list的简单模拟实现代码(常用接口) 

/
#pragma once
#include <iostream>
#include <algorithm>
#include <assert.h>
#include <list>
using namespace std;namespace ELY {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){}};template<class T,class Ref,class Ptr>struct list_iterator//tip{typedef list_node<T> Node;typedef list_iterator<T,Ref,Ptr> Self;Node* _node;list_iterator(Node* node):_node(node){}Ref operator*(){return _node->_data;}Ptr operator->(){return &_node->_data;}Self& operator++(){_node = _node->_next;return *this;}Self& operator--(){_node = _node->_prev;return *this;}Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}Self operator++(int){Self tmp(*this);_node = _node->_next;return tmp;}bool operator!=(const Self& s){return _node != s._node;}bool operator==(const Self& s){return _node == s._node;}};template<class T>class list{void empty_init(){_head = new Node();_head->_next = _head;_head->_prev = _head;}public:typedef list_node<T> Node;typedef list_iterator<T,T&,T*> iterator;//tiptypedef list_iterator<T,const T&,const T*> const_iterator;//tiplist(){empty_init();}list(const list<T>& list){empty_init();for (auto i : list){push_back(i);}}list<T>& operator=(list<T> list){swap(list);return *this;}list(size_t n, const T& val = T()){empty_init();for (size_t i = 0; i < n; i++){push_back(val);}}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 push_back(const T& x = T()){Node* newnode = new Node(x);Node* tail = _head->_prev;newnode->_next = _head;newnode->_prev = tail;tail->_next = newnode;_head->_prev = newnode;}iterator insert(iterator pos, const T& val)//tip{Node* newnode = new Node(val);Node* cur = pos._node;cur->_prev->_next = newnode;newnode->_prev = cur->_prev;cur->_prev = newnode;newnode->_next = cur;return iterator(newnode);}iterator push_front(const T& val){return insert(begin(), val);}iterator erase(iterator pos){assert(pos != end());Node* cur = pos._node;cur->_next->_prev = cur->_prev;cur->_prev->_next = cur->_next;pos = cur->_next;delete cur;return pos;}iterator pop_front(){return erase(begin());}iterator pop_back(){return erase(--end());}void clear(){auto i = begin();while (i != end()){i = erase(i);}}void swap(list<T>& list){std::swap(_head, list._head);}~list(){clear();delete _head;_head = nullptr;}private:Node* _head;};
}
/
mylist.h

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/451067.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络:数据链路层 —— 共享式以太网

文章目录 共享式以太网CSMA/CD 协议CSMA/CD 协议 的基本原理 共享式以太网的争用期共享式以太网的最小帧长共享式以太网的最大帧长共享式以太网的退避算法截断二进制指数退避算法 共享二进制以太网的信道利用率使用集线器的共享式以太网10BASE-T 共享式以太网 共享式以太网是当…

自监督学习:引领机器学习的新革命

引言 自监督学习&#xff08;Self-Supervised Learning&#xff09;近年来在机器学习领域取得了显著进展&#xff0c;成为人工智能研究的热门话题。不同于传统的监督学习和无监督学习&#xff0c;自监督学习通过利用未标注数据生成标签&#xff0c;从而大幅降低对人工标注数据…

Modbus TCP 西门子PLC指令以太口地址配置以及 Poll Slave调试软件地址配置

1前言 本篇文章讲了 Modbus TCP通讯中的一些以太网端口配置和遇到的一些问题&#xff0c; 都是肝货自己测试的QAQ。 2西门子 SERVER 指令 该指令是让外界设备主动连接此PLC被动连接&#xff0c; 所以这里应该填 外界设备的IP地址。 这边 我因为是电脑的Modbus Poll 主机来…

反弹shell检测的一些思路

前言 反弹shell是攻击者常用的手段之一&#xff0c;通过反弹Shell&#xff0c;攻击者可以绕过防火墙&#xff0c;获取目标系统的shell访问权限&#xff0c;进行后续的恶意操作。因此&#xff0c;及时检测并阻止反弹Shell行为对于安全防护来说非常重要。本文通过介绍反弹shell的…

Kafka原理剖析之「Purgatory(炼狱 | 时间轮)」

一、前言 本文介绍一下Kafka赫赫有名的组件Purgatory&#xff0c;相信做Kafka的朋友或多或少都对其有一定的了解&#xff0c;至少是听过它的名字。那它的作用是什么呢&#xff0c;用来解决什么问题呢&#xff1f;官网confluent早就有文章对其做了阐述 https://cwiki.apache.o…

Redis和Jedis的区别

目录 含义与用途 Jedis案例 总结 含义与用途 Redis&#xff1a; 概念&#xff1a;Redis是一个基于内存的键值存储数据库&#xff0c;支持丰富的数据结构。比如&#xff1a;字符串功能&#xff1a;除了基础的数据存储&#xff0c;Redis还提供了丰富的高级功能。如持久化&…

golang生成并分析cpu prof文件

1. 定义一个接口&#xff0c;请求接口时&#xff0c;生成cpu.prof文件 在主协程中新启一个协程&#xff0c;当请求接口时&#xff0c;生成一个60秒的cpu.prof文件 go func() {http.HandleFunc("/prof", startProfileHandler)http.ListenAndServe(":9092"…

MySQL中什么情况下类型转换会导致索引失效

文章目录 1. 问题引入2. 准备工作3. 案例分析3.1 正常情况3.2 发生了隐式类型转换的情况 4. MySQL隐式类型转换的规则4.1 案例引入4.2 MySQL 中隐式类型转换的规则4.3 验证 MySQL 隐式类型转换的规则 5. 总结 如果对 MySQL 索引不了解&#xff0c;可以看一下我的另一篇博文&…

markdown 笔记,语法,技巧

起因&#xff0c; 目的: markdown 有些语法&#xff0c;不常用&#xff0c;记不住。单独记录一下。 1. 插入数学公式 用 $$ 来包裹住多行数学公式。 $$ 多行数学公式 $$ 2. 2个星号 ** &#xff0c; 加粗&#xff0c; 3. 单行代码的 引用&#xff0c; 左右各一个顿号 8.…

HTML_文本标签

概念&#xff1a; 1、用于包裹&#xff1a;词汇、短语等。 2、通常写在排版标签里面。 3、排版标签更宏观(大段的文字)&#xff0c;文本标签更微观(词汇、短语)。 4、文本标签通常都是行内元素。 常用的文本标签 标签名 全称 标签语义em Emphasized 加重(文本)。要着重阅…

数字图像处理:图像复原应用

数字图像处理&#xff1a;图像复原应用 1.1 什么是图像复原&#xff1f; 图像复原是图像处理中的一个重要领域&#xff0c;旨在从退化&#xff08;例如噪声、模糊等&#xff09;图像中恢复出尽可能接近原始图像的结果。图像复原与图像增强不同&#xff0c;复原更多地依赖于图…

3D一览通常见问题QA

感谢大家一直以来对大腾智能3D一览通的支持&#xff0c;我们致力于提供便捷高效的3D协同服务。这里小编整理了一些关于3D一览通的常见问题&#xff0c;以便大家更好地了解和使用3D一览通。 Q&#xff1a;3D一览通的功能是什么&#xff1f; 3D一览通是大腾智能打造的一款云端轻…

如何在 JSON 中编写“anyOf”语句?

在 JSON 中&#xff0c;anyOf 语句通常用于 JSON Schema&#xff08;JSON 模式&#xff09;中&#xff0c;来定义多个可能的模式&#xff0c;表示数据可以匹配多个子模式中的任意一个。这种功能常用于验证 JSON 数据是否符合某一组可能的条件之一。 1、问题背景 问题&#xff…

【计算机网络 - 基础问题】每日 3 题(三十六)

✍个人博客&#xff1a;https://blog.csdn.net/Newin2020?typeblog &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞…

MongoDB 的安装详情

在虚拟机里面opt下 新建一个mongodb文件夹 再新建一个opt/mongodb/data文件夹&#xff0c; 然后将挂载的mongodb数据放到data文件夹里&#xff1a; 【把mongodb的数据挂载出来&#xff0c;以后我们再次重启的时候 数据起码还会在】 冒号右边 挂载到左边的路径 docker run -…

Matlab终于能够实现Transformer预测了

声明&#xff1a;文章是从本人公众号中复制而来&#xff0c;因此&#xff0c;想最新最快了解各类智能优化算法及其改进的朋友&#xff0c;可关注我的公众号&#xff1a;强盛机器学习&#xff0c;不定期会有很多免费代码分享~ 目录 原理简介 数据介绍 结果展示 完整代码 今…

ubuntu24 修改ip地址 ubuntu虚拟机修改静态ip

1. ubuntu 修改地址在/etc/netplan # 进入路径 cd /etc/netplan # 修改文件夹下的配置文件&#xff0c;我的是50-cloud-init.yaml. ye可能你得是20-cloud-init.yaml 2. 修改为&#xff1a; dhcp4: 改为false 192.168.164.50 是我自己分配的ip地址, /24 为固定写法&#xff…

数据结构与算法:堆与优先队列的深入剖析

数据结构与算法&#xff1a;堆与优先队列的深入剖析 堆是一种特殊的树形数据结构&#xff0c;广泛应用于优先队列的实现以及各种高效的算法中&#xff0c;如排序和图算法。通过深入了解堆的结构、不同堆的实现方式&#xff0c;以及堆在实际系统中的应用&#xff0c;我们可以掌…

初级网络工程师之从入门到入狱(四)

本文是我在学习过程中记录学习的点点滴滴&#xff0c;目的是为了学完之后巩固一下顺便也和大家分享一下&#xff0c;日后忘记了也可以方便快速的复习。 网络工程师从入门到入狱 前言一、Wlan应用实战1.1、拓扑图详解1.2、LSW11.3、AC11.4、抓包1.5、Tunnel隧道模式解析1.6、AP、…

服务器软件之Tomcat

服务器软件之Tomcat 服务器软件之Tomcat 服务器软件之Tomcat一、什么是Tomcat二、安装Tomcat1、前提&#xff1a;2、下载3、解压下载的tomcat4、tomcat启动常见错误4.1、tomcat8.0 startup报错java.util.logging.ErrorManager: 44.2、java.lang.UnsatisfiedLinkError 三、Tomca…