List中的迭代器实现【C++】

List中的迭代器实现【C++】

  • 一. list的结构
  • 二. 迭代器的区别
  • 三. 迭代器的实现
    • i. 类的设计
    • ii. ++重载
    • iii. !=重载
    • iiii. begin()
    • iiiii. end()
    • iiiii. operator*
  • 四.测试
  • 五. const迭代器的实现
    • i. 实现
    • 5.2 优化实现

一. list的结构

其实按照习惯来说,应该要专门出一篇博客来写

list的模拟实现,但是其实list与vector以及string的主要区别在于迭代器的设计。

所以就偷个懒直接写迭代器的部分了。

这里先声明以下,我这里就是进行一下模拟实现,STL中的list的iterator虽然并不是这样实现的,但是实现逻辑和结构都大差不差

这里就先贴上list类的结构:

namespace list
{
    template<class T>
    struct list_node
    {
        list_node<T>* _prev;
        list_node<T>* _next;
        T _val;
        //节点的构造函数
        list_node(const T& val = T())
            :_prev(nullptr)
            , _next(nullptr)
            , _val(val)
        {}
    };
    template<class T>
    class list
    {
    public:
        list()
        {
        //链表的默认构造
            list_node<T>* head = new list_node<T>;
            //初始化哨兵位
            _head = head;
            _head->_next = _head;
            _head->_prev = _head;
            _size = 0;
        }
        ~list()
        {
        }
    private:
        list_node<T>* _head;
    };
}

这个其实和我们之前写的双向带头循环链表一样。

二. 迭代器的区别

迭代器的区别,落到实质问题上
还是容器与容器的区别,也就是vector与list的区别

vector中的iterator是这样去实现的。

首先:

typedef T* iterator;

这里先重命名

iterator begin()
		{
			return _start;
		}
		iterator end()
		{
			return _finish;
		}

随后直接设计end与begin的函数。

之后就能发现,范围for与迭代器能直接进行使用了。

因为我们实现vector指针的方法是直接将内置类型指针进行改名。

而内置类型指针支持++,所以可以直接运行程序了。
但我们想想list肯定就不支持这样的操作了

vector能这这样,是因为数据存储在内存空间中连续时,正好可以进行使用。

但是list的问题是,内存空间是不连续的
这样的话再用指针内部支持的++算法就不太合适了。

所以我们今天就是为了来模拟实现list的迭代器的实现。

首先的目标就是实现下面这个代码的跑动。

	list::list<int>::iterator it = l1.begin();
	while (it != l1.end())
	{
		std::cout << *it;
		++it;
	}

三. 迭代器的实现

我们首先来看,我们最难解决的问题就是it的前置++
因为vector和list最重要的问题就是运算的方式不一样。

那我们想要用我们自己的方式进行++。
这个时候就应该想到了我们模拟实现类的时候最常用的东西:重载运算符

重载运算符在哪里能用:自定义类型
所以我们这个时候应该自然而然的想到自己写一个类,来充当iterator
从而实现我们想要的++方式。

i. 类的设计

    template<class T>
    struct __list_iterator
    {
        list_node<T>* _node;
        __list_iterator(list_node<T>* node)
            :_node(node)
        {}

    };

基本上大致结构是这样。

其中list_node就是双链表的节点结构。(上面写过)

这里添加个typedef

        template<class T>
        class list
        {
        public:
            typedef __list_iterator<T> iterator;

这个typedef和vefctor的使用没有什么大区别了。

现在来专注实现上面代码的所有的重载即可

ii. ++重载

  __list_iterator<T>& operator++()
  {
      _node = _node->_next;
      return *this;
  }

这个就是我们以前写的双链表和单链表的部分了。

iii. !=重载

 bool operator!=(const __list_iterator<T>& node)
 {
     return _node != node._node;
 }

iiii. begin()

注意:这里的end和begin都是在list中的

因为迭代器执行时,会在list中寻找begin和end

 iterator begin()
 {
     return iterator(_head->_next);
 }

这里我们需要返回的是迭代器的类型

所以需要调用一下迭代器的构造函数

iiiii. end()

 iterator end()
 {
     return _head;
 }

我们发现begin需要调用构造函数

但是这边end却没有调用

因为单参数的构造函数返回时
如果返回的类型和需要返回的类型不同
就会主动调用需要返回类型的构造函数进行转换

iiiii. operator*

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

四.测试

这里就直接写一个push_back的方法

  void push_back(const T& val)
  {
      insert(end(), val);
  }
        void insert(iterator pos, const T& val)
        {
            list_node<T>* new_node = new list_node<T>(val);
            _size++;
            pos._node->_prev->_next = new_node;
            new_node->_prev = pos._node->_prev;
            new_node->_next = pos._node;
            pos._node->_prev = new_node;
        }

接下来进行测试即可

	list::list<int> l1;
	l1.push_back(1);
	l1.push_back(1);
	l1.push_back(1);
	l1.push_back(1);
	l1.push_back(1);
	list::list<int>::iterator it = l1.begin();
	while (it != l1.end())
	{
		std::cout << *it;
		++it;
	}

在这里插入图片描述
这里能发现程序完美执行了。

五. const迭代器的实现

我们在使用STL中自带的迭代器时

应该经常能用到const迭代器。

const_iterator

这里我们首先要清楚const_iterator实现的是什么:

我们清楚效果是:指向的内容不能修改,但是迭代器本身可以修改。

所以实现的类型就是const T* ptr

而不是T* const ptr

那我们要达成这种效果。
可以从函数的返回值上入手

平常使用函数时,基本上都是通过重载符*
来进行对应值的修改

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

那我们这样不就可以了吗。。

i. 实现

typedef __list_iterator<T>const_iterator;

这里在list中重命名一下。

    template<class T>
    struct __list_const_iterator
    {
        list_node<T>* _node;
        __list_const_iterator(list_node<T>* node)
            :_node(node)
        {}
      

        const T& operator*()
        {
            return _node->_val;
        }
        __list_const_iterator<T>& operator++()
        {
            _node = _node->_next;
            return *this;
        }
     
      

        bool operator!=(const  __list_const_iterator<T>& node)
        {
            return _node != node._node;
        }
    };

之后再把这个类丢进去。

但是这样会发现,实现的太过繁杂了。

这里就来个优化版本。

5.2 优化实现

这里先直接上结果

list中的重命名:

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

具体实现:

    template<class T, class ref>
    struct __list_iterator
    {
        list_node<T>* _node;
        __list_iterator(list_node<T>* node)
            :_node(node)
        {}
      

        ref& operator*()
        {
            return _node->_val;
        }
        __list_iterator<T, ref, ptr>& operator++()
        {
            _node = _node->_next;
            return *this;
        }
     
      

        bool operator!=(const  __list_iterator<T, ref, ptr>& node)
        {
            return _node != node._node;
        }
    };

这里是给模板新添加了一个参数。

从而实现const与普通的两种类型迭代器的实现。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2.1 Windows驱动开发:内核链表与结构体

在Windows内核中&#xff0c;为了实现高效的数据结构操作&#xff0c;通常会使用链表和结构体相结合的方式进行数据存储和操作。内核提供了一个专门用于链表操作的数据结构LIST_ENTRY&#xff0c;可以用来描述一个链表中的每一个节点。 使用链表来存储结构体时&#xff0c;需要…

微信小程序:仅前端实现对象数组的模糊查询

效果 核心代码 //对数组进行过滤&#xff0c;返回数组中每一想满足name值包括变量query的 let result array.filter(item > { return item.name.includes(query); }); 完整代码 wxml <input type"text" placeholder"请输入名称" placeholder-styl…

【华为OD题库-015】报文重排序-Java

题目 对报文进行重传和重排序是常用的可靠性机制&#xff0c;重传缓冲区内有一定数量的子报文&#xff0c;每个子报文在原始报文中的顺序已知&#xff0c;现在需要恢复出原始报文。 输入描述 输入第一行为N,表示子报文的个数&#xff0c;0<N < 1000。 输入第二行为N个子报…

安防监控EasyCVR视频汇聚平台运维现场无法使用Linux抓包该如何解决?

视频云存储/安防监控EasyCVR视频汇聚平台基于云边端智能协同&#xff0c;支持海量视频的轻量化接入与汇聚、转码与处理、全网智能分发、视频集中存储等。监控视频平台EasyCVR拓展性强&#xff0c;视频能力丰富&#xff0c;具体可实现视频监控直播、视频轮播、视频录像、云存储、…

竞赛选题 深度学习的口罩佩戴检测 - opencv 卷积神经网络 机器视觉 深度学习

文章目录 0 简介1 课题背景&#x1f6a9; 2 口罩佩戴算法实现2.1 YOLO 模型概览2.2 YOLOv32.3 YOLO 口罩佩戴检测实现数据集 2.4 实现代码2.5 检测效果 3 口罩佩戴检测算法评价指标3.1 准确率&#xff08;Accuracy&#xff09;3.2 精确率(Precision)和召回率(Recall)3.3 平均精…

“大学生”返乡投身乡村建设,直播电商成为返乡创业新潮流!

数字乡村建设是新时代乡村振兴的必经之路&#xff0c;它是伴随网络化、信息化和数字化在农业农村经济社会发展中的应用&#xff0c;以及农民现代信息技能的提高而内生的农业农村现代化发展和转型进程&#xff0c;既是乡村振兴的战略方向&#xff0c;也是建设数字中国的重要内容…

vue3使用时遇到的问题

使用elementplus遇到的问题 1.el-form中input无法输入 问题描述&#xff1a;在el-form中的el-input中输入数字或字母时出现卡顿&#xff0c;输入不进去的现象 问题原因&#xff1a;el-form的ref和model的名称写成了一样的单词 问题解决&#xff1a;两个不能一样 2.input去除…

CTFhub-RCE-php://input

我们需要使用php://input来构造发送的指令 查看phpinfo&#xff0c;找到一下字段 证明是可以使用php://input 1. 使用Burpsuite抓包并转至Repeater 2. 构造包 方法&#xff1a;POST 目标&#xff1a;/?filephp://input Body&#xff1a;<?php system("ls /"…

约束条件的安全测试_报错注入

约束条件的安全测试_报错注入 基于约束的SQL攻击 报错注入

(附源码)基于SSM旅行社网站-计算机毕设 90030

SSM旅行社网站 摘 要 旅游业是一个信息密集型产业&#xff0c;传统的旅游景点门票售卖受到技术和人力的限制&#xff0c;旅行社网站则可以建立景区与游客之间的有效通道&#xff0c;能更好的满足游客便捷旅游的需求。旅行社网站的设计是基于SSM框架、Mysql数据库、JSP技术、Aja…

wireshark打开tcpdump抓的包 vwr: Invalid data length runs past the end of the record

tcpdump -i any -n -s0 > t.pcap 使用此命令在Debian系统上抓包&#xff0c;下载到PC&#xff0c;用wireshark打开时报错&#xff1a; 后来发现写入文件时使用 -w 是没问题的&#xff0c;原因还不清楚。 tcpdump -i any -n -s0 -w t.pcap

Mysql-数据类型

1.数据类型分类 2. 整形类型 说明 : 在MySQL中&#xff0c;整型可以指定是有符号的和无符号的&#xff0c;默认是有符号的。可以通过UNSIGNED来说明某个字段是无符号的。 注意&#xff1a;尽量不使用unsigned&#xff0c;对于int类型可能存放不下的数据&#xff0c;int unsign…

7个好用的可视化数据平台,让你的数据分析更高效率、高逼格

在信息爆炸的时代&#xff0c;数据是企业决策的重要依据。为了更高效率、更高逼格地进行数据分析&#xff0c;选择一个优秀的可视化数据平台至关重要。在众多可选项中&#xff0c;VeryReport报表软件脱颖而出&#xff0c;成为最好用的可视化数据平台之一&#xff0c;以下是其突…

Pyside6/PYQT6如何实现无边框设计,解决无边框窗口无法移动和实现窗口拖拽改变大小的问题

文章目录 💢 问题 💢💯 解决方案 💯🍔 准备工作📚 setWindowFlags、setWindowFlag和setAttribute的区别🐾 操作步骤🐾 窗口无边框🐾 窗口透明🐾 实现窗口可移动🐾 实现窗口拖拽改变大小⚓️ 相关链接 ⚓️💢 问题 💢 有时候我们需要一个无边框的UI设…

活跃类指标

活跃类指标反映了用户的真实使用情况。本节我们深入探讨活跃类指标的核心逻辑。 1&#xff0e; UV UV ( Unique Visitor &#xff0c;独立访客&#xff09;&#xff0c;是所有活跃类指标的基础。 既然叫独立访客&#xff0c;何谓之独立&#xff1f; APP 产品界定独立访客相对…

如何用postman+jmeter实现接口实例

一、接口基础 为什么要单独测试接口&#xff1f; 1. 程序是分开开发的&#xff0c;前端还没有开发&#xff0c;后端已经开发完了&#xff0c;可以提前进入测试 2. 接口直接返回的数据------越底层发现bug&#xff0c;修复成本是越低的 3. 接口测试能模拟功能测试不能测到的异…

2.2 Windows驱动开发:内核自旋锁结构

提到自旋锁那就必须要说链表&#xff0c;在上一篇《内核中的链表与结构体》文章中简单实用链表结构来存储进程信息列表&#xff0c;相信读者应该已经理解了内核链表的基本使用&#xff0c;本篇文章将讲解自旋锁的简单应用&#xff0c;自旋锁是为了解决内核链表读写时存在线程同…

医院等级评审,离不开医院不良事件报告系统

医院不良事件报告系统全套源码 不良事件管理系统源码 不良事件上报系统对事件的报告、处置、跟踪、评价、分析、改进、学习等进行了综合管理&#xff0c;通过双向互评机制实现临床科室与职能部门之间的进一步互动&#xff0c;加强不良事件报告处置过程中的信息互通能力。 围绕…

项目生命周期分享

第一阶段&#xff1a; 项目启动&#xff0c;2天时间即可&#xff0c;需要输出项目进度计划 1.项目组成立1天&#xff0c;用来建立项目组&#xff0c;确定工作分工和工作方法&#xff0c;指定项目总体计划&#xff08;包括前期交流&#xff0c;需求收集&#xff0c;项目立项等…

数组区域检索的优化 --- 分块,线段树,树状数组

思考 首先让我们来思考一个问题&#xff0c;给定一个数组&#xff0c;和left与right的值&#xff0c;让你求这个数组中left到right之间元素的和&#xff0c;你会怎么计算&#xff1f;最简单的当然是遍历。如果有人问你这个问题的时候&#xff0c;他决对是会让你优化的&#xff…