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

反向迭代器:reverse_iterator的实现

目录

前言

特点

注意事项

实现

构造函数

功能函数

在list与vector中的使用

vector

list


前言

反向迭代器是一种在序列容器的末尾开始,并向前移动至序列开始处的迭代器。在C++中,反向迭代器由标准库中的容器类提供,比如vectorlistdeque等。它们允许程序员以逆序遍历容器中的元素。

以下是反向迭代器的一些特点:

特点

  1. 反向遍历:反向迭代器从序列的最后一个元素开始,逐步移动到第一个元素。
  2. 操作符重载:C++中的反向迭代器重载了递增(++)和递减(--操作符。递增操作使迭代器向序列的开始方向移动,而递减操作使其向序列的结束方向移动。
  3. 类型:反向迭代器的类型通常由容器类型加上reverse_iterator后缀表示,例如vector<int>::reverse_iterator

注意事项

  • 反向迭代器不支持所有的普通迭代器的操作,例如算术操作(加、减)。
  • 使用反向迭代器时,递增操作实际上是向序列的开始方向移动。
  • 在C++标准库中,并不是所有容器都支持反向迭代器。只有那些支持双向迭代器(BidirectionalIterator)或随机访问迭代器(RandomAccessIterator的容器才提供反向迭代器。这是因为反向迭代器需要能够向前和向后遍历容器,而这两种迭代器都支持这些操作。

需要逆序访问容器元素时,它们可以简化代码并提高效率

反向迭代器有const版本和非const版本,所以我们需要实现两个版本。

实现

反向迭代器由于与正向迭代器的行为相似,因此借鉴适配器的思想,用正向迭代器实现反向迭代器

同时我们增加两个模板参数Ref Ptr,作为const T&和Const T*的区分


template<class Iterator, class Ref, class Ptr>
class ReverseIterator

成员变量就是一个被适配的正向迭代器

private:
    Iterator _it;

构造函数

	ReverseIterator(Iterator it):_it(it){}

用传入的模板迭代器去初始化成员

功能函数

++ --

Self& operator++(){--_it;return *this;}Self& operator--(){++_it;return *this;}

* ->两种解引用

为了实现对称,解引用时,解引用的是当前位置的下一个数据

Ref operator*()	//内部去调用普通迭代器的解引用{Iterator cur = _it;return *(--cur);	//返回数据的引用,不能--_it,防止迭代器错位}//->也是一种解引用Ptr operator->()	//返回的其实是一个指针{return &(operator*());}

==    !=

迭代器的比较,看看成员参数是不是一个

bool operator!=(const Self& s){return _it != s._it;}bool operator==(const Self& s){return _it == s._it;}

在list与vector中的使用

vector

首先我们需要展开头文件

#include "reverse_iterator.h"    //反向迭代器头文件在此展开
 

然后利用typedef将迭代器重命名

typedef T* iterator;
typedef const T* const_iterator;
typedef Reverse_Iterator<iterator, T&, T*> reverse_iterator;
typedef Reverse_Iterator<const_iterator, const T&, const T*> const_reverse_iterator;

rbegin与rend

reverse_iterator rbegin()
{return reverse_iterator(end());		
}reverse_iterator rend()
{return reverse_iterator(begin());
}const_reverse_iterator rbegin() const
{return const_reverse_iterator(end());
}const_reverse_iterator rend() const
{return const_reverse_iterator(begin());
}

其中const反向迭代器将调用const成员(end()、begin())

((((

题外话:

返回时,采用的是传值返回

自定义类型的传值返回通常是通过拷贝构造函数来实现的。当一个对象作为函数的返回值时,如果采用值返回的方式,函数内部会创建一个临时对象,这个临时对象是通过拷贝构造函数来初始化的,它是对返回对象的一个副本。这个副本具有常性(体现在引用时)

这里一个特例,这是一个匿名对象,具有常性,但是却可以调用非const成员函数。

示例:

class MyClass {
public:MyClass() {// 构造函数}MyClass(const MyClass& other) {// 拷贝构造函数}// 其他成员函数和成员变量...
};MyClass createObject() {MyClass obj;// 对obj进行一些操作return obj; // 这里会调用拷贝构造函数来构造返回值
}int main() {MyClass result = createObject(); // 接收返回值,同样会调用拷贝构造函数return 0;
}

在上述代码中,当createObject函数返回obj时,会调用MyClass的拷贝构造函数来构造一个临时对象,这个临时对象随后会被用来初始化main函数中的result对象。因此,在这个过程中至少会发生两次拷贝构造:一次是在函数返回时构造临时对象,另一次是在接收返回值时。

需要注意的是,现代编译器通常会对此类操作进行优化,比如返回值优化(NRVO,Named Return Value Optimization)或者拷贝省略(copy elision),从而避免不必要的拷贝,以提高性能。在C++11及以后的版本中,这种优化是被标准所允许的,甚至在某些情况下是强制的。

示例2:

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

产生临时对象后:

这个临时对象会在表达式结束时被销毁,但是因为这是在一个返回语句中,所以返回的对象会被用来初始化函数调用的结果。这样,当函数调用者接收这个返回值时,他们实际上是在接收一个复制构造的 iterator 临时对象,而不是原始的返回对象。接收时,会使用拷贝构造等手段完成接收。

))))

list

同样得展开头文件

#include "reverse_iterator.h"    //对应头文件内容在此展开

typedef __list_iterator<T, T&, T*> iterator;
typedef __list_iterator<T, const T&, const T*> const_iterator;
typedef Reverse_Iterator<iterator, T&, T*> reverse_iterator;	//把我的迭代器传给这个全局的类
typedef Reverse_Iterator<const_iterator, const T&, const T*> const_reverse_iterator;

Reverse_Iterator这个模板类已经在此文件中展开,因此在list类中可以直接使用这个类模板,并借助自身的成员去实例化这个模板类。 

reverse_iterator rbegin() {						//强调对称return reverse_iterator(end());		//借助end()迭代器构造反向迭代器}reverse_iterator rend(){return reverse_iterator(begin());}const_reverse_iterator rbegin() const	//const迭代器{return const_reverse_iterator(end());	//调用的是const函数end()}const_reverse_iterator rend() const{return const_reverse_iterator(begin());	}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 嵌入式Linux C应用编程指南-高级I/O(速记版)
  • 科研绘图系列:R语言组合图形绘图
  • unity游戏开放:标记物体 一目了然
  • Bean 的生命周期
  • gpt开发
  • vue如何引入element-ui
  • 有了它 一键掌握Vue新版本!
  • 【学习笔记】卫星通信NTN 3GPP标准化进展分析(六)- 参考标准
  • 基于Java的基础简单网络编程
  • Redis从入门再到入门(中)
  • 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch12 随机森林(Random Forest)
  • 算法的学习笔记—把数字翻译成字符串
  • 23. 如何使用Collections.synchronizedList()方法来创建线程安全的集合?有哪些注意事项?
  • 【数据结构初阶】二叉树--堆(顺序结构实现)
  • Linux——命令行文件的管理(创建,复制,删除,移动文件,硬链接与软链接)
  • [译]如何构建服务器端web组件,为何要构建?
  • 【个人向】《HTTP图解》阅后小结
  • 【跃迁之路】【669天】程序员高效学习方法论探索系列(实验阶段426-2018.12.13)...
  • 77. Combinations
  • android图片蒙层
  • FastReport在线报表设计器工作原理
  • HTML-表单
  • Java 23种设计模式 之单例模式 7种实现方式
  • js算法-归并排序(merge_sort)
  • Laravel 中的一个后期静态绑定
  • open-falcon 开发笔记(一):从零开始搭建虚拟服务器和监测环境
  • Terraform入门 - 3. 变更基础设施
  • v-if和v-for连用出现的问题
  • vue从创建到完整的饿了么(18)购物车详细信息的展示与删除
  • win10下安装mysql5.7
  • 订阅Forge Viewer所有的事件
  • 给第三方使用接口的 URL 签名实现
  • 规范化安全开发 KOA 手脚架
  • 看图轻松理解数据结构与算法系列(基于数组的栈)
  • 如何设计一个比特币钱包服务
  • 双管齐下,VMware的容器新战略
  • 策略 : 一文教你成为人工智能(AI)领域专家
  • 扩展资源服务器解决oauth2 性能瓶颈
  • ​​​【收录 Hello 算法】9.4 小结
  • ![CDATA[ ]] 是什么东东
  • ###STL(标准模板库)
  • #我与Java虚拟机的故事#连载09:面试大厂逃不过的JVM
  • $con= MySQL有关填空题_2015年计算机二级考试《MySQL》提高练习题(10)
  • (M)unity2D敌人的创建、人物属性设置,遇敌掉血
  • (定时器/计数器)中断系统(详解与使用)
  • (二)hibernate配置管理
  • (附源码)计算机毕业设计ssm基于Internet快递柜管理系统
  • (论文阅读23/100)Hierarchical Convolutional Features for Visual Tracking
  • (十二)springboot实战——SSE服务推送事件案例实现
  • (四)activit5.23.0修复跟踪高亮显示BUG
  • (一) storm的集群安装与配置
  • (已解决)vue+element-ui实现个人中心,仿照原神
  • (转) SpringBoot:使用spring-boot-devtools进行热部署以及不生效的问题解决
  • (转)甲方乙方——赵民谈找工作
  • (转)使用VMware vSphere标准交换机设置网络连接