【C++】list模拟实现list迭代器失效问题

list模拟实现&list迭代器失效问题

  • 一,list模拟实现
    • 1. list的主要框架接口模拟
    • 2. list构造&拷贝构造&析构
    • 3. list迭代器
      • 3.1 普通迭代器
      • 3.2 const迭代器
    • 4. 增删查改
  • 二,迭代器失效问题
    • 1. list的迭代器失效原因
    • 2. 解决办法

一,list模拟实现

在上节我们熟悉了list的底层结构和各个接口的含义后,下面我们来对list的主要接口进行模拟实现

1. list的主要框架接口模拟

list的底层是双向带头循环链表,所以我们先写一个节点的类,这个节点类也是一个类模板,由给定的数据类型生成相应的类

这个节点类可以写成结构体,用结构体是为了方便访问数据,不用单独写取数据接口

//list的节点
template<class T>
struct ListNode {ListNode* _prev;ListNode* _next;T _data;ListNode(const T& t = T()):_prev(nullptr),_next(nullptr),_data(t){}};

其次我们写list的基本框架,其成员变量我们可以用节点声明一个头结点,根据头结点的链接关系我们就可以知道其list存放的内容

template<class T>
class mylist {typedef ListNode<T> Node;//方便阅读
public://....
private:Node* _head;
};

2. list构造&拷贝构造&析构

list的构造有无参的构造,用n个值初始化构造,迭代器区间构造


我们这里实现无参的构造

void list_init() {_head = new Node;_head->_next = _head;_head->_prev = _head;
}mylist() {list_init();
}

初始化时其头指针和外指针指向自己:
在这里插入图片描述


现在我们来实现拷贝构造

拷贝构造的实现我们和vector一样依次去插入即可。

mylist(mylist<T>& ml) {list_init();for (const auto e : ml) {push_back(e);}
}

重载operator=,这里我们可以转变一下思路,当传入一个list对象时,发生了拷贝构造,用拷贝构造的这个对象和当前要调用operator=的对象进行交换,再返回交换后的这个对象。

mylist& operator=(mylist lt)//在类中时可以省略模板参数swap(lt);return *this;
}

3. list迭代器

由于list的底层空间的不连续性,所以不能像string和vector一样直接用原生指针,我们要对其进行封装,使其可以像指针那样可以进行++或者–那样去使用。

template<class T>
struct List_Iterator {typedef ListNode<T> Node;typedef List_Iterator<T> self;Node* _node;//..
};

对于迭代器的构造函数,比较简单,可写可不写

template<class T>
struct List_Iterator {typedef ListNode<T> Node;typedef List_Iterator<T> self;Node* _node;List_Iterator(Node* n):_node(n){}//..
};

3.1 普通迭代器

我们先实现普通迭代器
既然要对这个类进行封装,那么就是为了让其可以进行++或者–等运算,所以我们需要对这些操作进行重载


重载operator++,有前置++和后置++,

//前置++
self& operator++() {_node = _node->_next;return  *this;
}//后置++
self operator++(int) {self tmp(*this);//拷贝构造_node = _node->_next;return tmp;
}

注:拷贝构造我们后面会实现


重载operator--和operator++类似,我们直接上代码

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

重载operator->,这是因为当list中存储的是其他结构体时,可以通过 it->_data 来直接访问其内容。

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

这里来举个具体的场景,看下面的代码:

struct Exp {int _a1 = 1;int _a2 = 2;Exp(int a1 = 1,int a2 = 2):_a1(a1),_a2(a2){}
};void test() {mylist<Exp> ex;ex.push_back(Exp(1, 2));mylist<Exp>::iterator it = ex.begin();while (it != ex.end()) {cout << (*it)._a1 << " " << (*it)._a2 << endl;cout << it->_a1 << " " << it->_a2 << endl;++it;}
}

看到这段代码,你可能会一头雾水,为啥 it->_a1可以直接访问struct的成员变量,it->不是返回的是这个结构体的指针吗?

在这里插入图片描述
其实这里是省略了一个->,我们可以加一句代码来解释

struct Exp {int _a1 = 1;int _a2 = 2;Exp(int a1 = 1,int a2 = 2):_a1(a1),_a2(a2){}
};void test() {mylist<Exp> ex;ex.push_back(Exp(1, 2));mylist<Exp>::iterator it = ex.begin();while (it != ex.end()) {cout << (*it)._a1 << " " << (*it)._a2 << endl;cout << it->_a1 << " " << it->_a2 << endl;cout << it.operator->()->_a1 << it.operator->()->_a2 << endl;++it;}
}

it->_a1中,it->取到的是结构体对象的地址,其后面应该再用一个->访问到其中的变量,但是后面的箭头被省略了,这里也是C++为了方便使用而做的处理。


operator*返回的是T& 减少拷贝

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

3.2 const迭代器

实现了普通迭代器,现在我们终于要实现const迭代器了,现在思考一下,普通迭代器和const迭代器区别在哪,其实const迭代器的区别就是operator*和operator->的返回值加上const。

试想一下,如果我们像vector一样只在普通迭代器前面直接加一个const。我们直接再将普通迭代器的代码拷贝一份,在operator*和operator->返回值前加上const,其余代码不变,也可以实现const迭代器。但是这样写代码难免有些冗余,如果相同的场景下要拷贝的代码有上千万行呢,难道也要拷贝吗?
在这里插入图片描述

所以这里我们只需在模板参数中添加两个参数Ref,Ptr。分别代表 operator和operator->的返回值,当传入的是const T或者const T&时,迭代器类会返回相应的类型。具体在list的类中再对const迭代器进行封装。

template<class T,class Ref,class Ptr>
struct List_Iterator {//typedef ListNode<T> Node;typedef List_Iterator<T,Ref,Ptr> self;Node* _node;List_Iterator(Node* n):_node(n){}//前置++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;}Ref operator* () {return _node->_data;}Ptr operator->() {return &_node->_data;}bool operator==(const self& s) {return _node == s._node;}bool operator!=(const self& s) {return _node != s._node;}};template<class T>
class mylist {typedef ListNode<T> Node;//方便阅读public:typedef List_Iterator<T,T&,T*> iterator;//写成公有,类外也可以访问typedef List_Iterator<T,const T&,const T*> const_iterator;//....
}

在list类中如何用用到iterator类呢,下面我们来实现一下:

iterator begin() {return _head->_next;//单参数的构造函数支持隐式类型转换,
}iterator end() {return _head;//迭代器区间是左闭右开,end指向最后一个元素的下一个位置
}

4. 增删查改

list的插入删除比较简单,因为不用考虑数据挪动的问题,只要创建或者删除新的节点,改变前后节点的指针即可。


insert

iterator insert(iterator pos,const T& t) {Node* newnode = new Node(t);Node* cur = pos._node;Node* prev = cur->_prev;prev->_next = newnode;newnode->_prev = prev;newnode->_next = cur;cur->_prev = newnode;return newnode;
}

erase可能会造成迭代器失效的问题,具体原因和解决办法我们在下面讲解。

iterator erase(iterator pos) {assert(pos != end());//会发生隐式类型转换,将指针类型转换成iterator类Node* cur = pos._node;Node* prev = cur->_prev;Node* next = cur->_next;prev->_next = next;next->_prev = prev;delete cur;cur = nullptr;return next;//迭代器失效,返回删除位置的下一个位置
}

push_back()和push_front和pop_back()和pop_front的实现可以直接复用insert,在末尾位置插入即可

void push_back(const T& t) {insert(end(), t);
}void push_front(const T& t) {insert(begin(), t);
}
void pop_back() {erase(--end());
}void pop_front() {erase(begin());
}

二,迭代器失效问题

1. list的迭代器失效原因

list的插入不会像vector造成迭代器失效,因为vector一旦发生扩容,其所有位置的迭代器都会失效,但是list不用扩容,当有元素插入时才创建节点,再链接到链表中。
但是list的删除会造成迭代器失效,因为删除一个节点后,这个节点的迭代器位置实际上已经不存在了,再继续使用则会出错,除非下次使用时再给其赋值。

2. 解决办法

解决办法就是让删除后返回被删除节点的下一个节点的迭代器,下面是正确的使用场景,

void Test()
{int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };list<int> l(array, array+sizeof(array)/sizeof(array[0]));auto it = l.begin();while (it != l.end()){l.erase(it++); // it = l.erase(it);}
}

list的模拟实现部分我们讲解完了,希望大家看了后会有所收获,期待更多的C++相关的知识请大家三连一波哦
在这里插入图片描述

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

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

相关文章

Java 汇编源码查看环境搭建

目录 一、简介 二、在IDEA开发环境中搭建汇编环境 2.1 在IDEA中搭建字节码查看环境 2.1.1 搭建步骤 2.1.1.1 第一步 2.1.1.2 第二步 2.1.1.3 第三步 2.1.1.4 第四步 2.1.2 验证 2.2 在IDEA开发环境中搭建汇编代码查看环境 2.2.2 配置HSDIS插件 2.2.3 验证HSDIS插件是…

js随机整数

在JavaScript中&#xff0c;您可以使用 Math.random() 函数生成一个0到1之间的随机数&#xff08;包括0&#xff0c;但不包括1&#xff09;&#xff0c;然后通过适当的缩放和取整&#xff0c;可以得到一个随机整数。以下是一个简单的函数&#xff0c;用于生成指定范围内的随机整…

[虚拟机保护逆向] [HGAME 2023 week4]vm

[虚拟机保护逆向] [HGAME 2023 week4]vm 虚拟机逆向的注意点&#xff1a;具体每个函数的功能&#xff0c;和其对应的硬件编码的*长度* 和 *含义*&#xff0c;都分析出来后就可以编写脚本将题目的opcode转化位vm实际执行的指令 &#xff1a;分析完成函数功能后就可以编写脚本输出…

概率论与数理统计 P6 条件概率

文章目录 P6 条件概率一.条件概率二.乘法定理三.全概率公式 & 贝叶斯公式3.1 全概率公式&#xff08;由因求果&#xff09;3.2 贝叶斯公式&#xff08;由果导因&#xff09; P6 条件概率 一.条件概率 1.Def&#xff1a;设A、B是两个事件&#xff0c;且 P ( A ) > 0 P(…

深度学习在硬件和计算平台上的优化:实现更快、更高效的突破

引言 深度学习&#xff0c;作为机器学习领域的一个子集&#xff0c;通过模拟人脑神经元的连接方式&#xff0c;构建复杂的网络结构来处理和分析数据。然而&#xff0c;随着深度学习模型规模的不断扩大和复杂度的提高&#xff0c;其对计算资源的需求也呈指数级增长。因此&#…

【MySQL】表的增删改查——MySQL基本查询、数据库表的创建、表的读取、表的更新、表的删除

文章目录 MySQL表的增删查改1. Create&#xff08;创建&#xff09;1.1 单行插入1.2 多行插入1.3 替换 2. Retrieve&#xff08;读取&#xff09;2.1 select查看2.2 where条件2.3 结果排序2.4 筛选分页结果 3. Update&#xff08;更新&#xff09;3.1 更新单个数据3.2 更新多个…

如何保证消息的可靠传输

数据的丢失问题&#xff0c;可能出现在生产者、MQ、消费者中 生产者丢失&#xff1a; 生产者将数据发送到 RabbitMQ 的时候&#xff0c;可能数据就在半路给搞丢了&#xff0c;因为网络问题啥的&#xff0c;都有可能。此时可以选择用 RabbitMQ 提供的事务功能&#xff0c;就是生…

Unmanaged PowerShell

简介 在渗透测试当中经常会使用到PowerShell来执行脚本, 但是直接使用PowerShell.exe是一个非常敏感的行为, EDR等产品对PowerShell.exe进程的创建监控的很密切, 并且随着PowerShell的渗透测试工具的普及, 越来越多的EDR会利用微软提供的AMSI接口对PS脚本进行扫描, 但是对于低…

vue实现购物车功能

实现功能 CSS部分 <style>.tr {display: flex;}.th {margin: 10px;width: 20%;height: 50%;}.td {display: flex;margin: 10px;width: 20%;height: 100px;align-items: center;}.app-container .banner-box {border-radius: 20px;overflow: hidden;margin-bottom: 10px;}…

input中文输入法导致的高频事件

这是基本结构 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>中文输入法的高频事件</title&…

以题为例 浅谈前缀和算法

前缀求和算法是什么 前缀和算法就是以空间去换取时间&#xff0c;可用于快速求数组的区间和&#xff0c;它可以用于一维数组和二维数组&#xff0c;但我现在只接触了一维数组并没有接触二维数组&#xff0c;所以在这里先介绍一维数组前缀和相关的知识 前缀和典型代码 for(int…

通信-CAN-01 总线拓扑

本文主要介绍CAN总线拓扑&#xff0c;并结合实际用到CAN设备做些说明。 1 总线拓扑 拓扑结构中分为CPU&#xff0c;CAN 控制器&#xff0c;收发器&#xff0c;双绞线。CAN控制器根据两根线上的电位差来判断总线电平。发送方通过使总线发生变化&#xff0c;将消息发送给接收方…

BPSK调制解调

BPSK数字调制是相移键控PSK的一种&#xff0c;通过数字信号&#xff0c;调制载波的相位&#xff0c;利用载波的相位变化来反映数字信号&#xff0c;载波的振幅和频率均不变化。PSK应用很广泛&#xff0c;抗噪声性能比ASK和FSK要好&#xff0c;频带利用率较高。BPSK中&#xff0…

探讨:C#运行程序文件并获取输出的小问题

尝试了一下用C#运行一个控制台程序&#xff0c;希望获取输出并在适当的时候输入&#xff0c;看起来挺简单&#xff0c;不过实际使用发现只能让程序一次性执行完毕并获取输出&#xff08;并没有仔细尝试各种方式&#xff09;。 代码很简单&#xff1a; private void Test(){Proc…

Java实战:Spring Boot 利用 Redis 解决海量重复提交问题

本文将详细介绍如何在 Spring Boot 应用程序中利用 Redis 解决海量重复提交问题。我们将深入探讨重复提交问题的原因和影响&#xff0c;以及如何使用 Redis 的数据结构和原子操作来控制请求的重复提交。 1. 引言 在现代的互联网应用中&#xff0c;用户可能会频繁地提交相同的…

前端文件流、切片下载和上传

前端文件流、切片下载和上传技术在提升文件传输效率和优化用户体验方面发挥着关键作用。这些技术不仅可以帮助解决大文件传输过程中可能遇到的问题&#xff0c;如网络超时、内存溢出等&#xff0c;还能通过并行传输和断点续传等功能&#xff0c;提高传输速度和稳定性。 一、前端…

C++高级面试提:请解释 C++ 中的指针算术和指针迭代器的区别。

请解释 C 中的指针算术和指针迭代器的区别。 在 C 中&#xff0c;指针算术&#xff08;Pointer Arithmetic&#xff09;和指针迭代器&#xff08;Pointer Iterators&#xff09;是处理指针的两种不同方法。 指针算术&#xff08;Pointer Arithmetic&#xff09;&#xff1a; …

webpack5基础--07_自动清空上次打包资源

自动清空上次打包资源 1. 配置 const path require("path");module.exports {entry: "./src/main.js",output: {path: path.resolve(__dirname, "dist"),filename: "static/js/main.js",clean: true, // 自动将上次打包目录资源清…

每日学习笔记:C++ 11的Tuple

#include <tuple> Tuple介绍(不定数的值组--可理解为pair的升级版) 定义 创建 取值 初始化 获取tuple元素个数、获取tuple某元素类型、将2个tuple类型串接为1个新tuple类型

解决Ubuntu 16.04/18.04 图形化界面异常、鼠标光标消失、鼠标变成叉叉等问题

bug场景&#xff1a; 一切从一次换源说起…叭叭叭 这篇文章解决的问题&#xff1a; 1.换源&#xff0c;默认源太慢&#xff0c;换成可用的阿里云的源 2.apt-get failed to …问题 3.图形化异常问题 4.get unmet dependence 问题 5. 鼠标光标消失和鼠标变成叉叉问题。 解决方…