反向迭代器: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());	}

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

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

相关文章

Qt 字符串的编码方式,以及反斜杠加3个数字是什么编码\344\275\240,如何生成

Qt 字符串的编码方式 问题 总所周知&#xff0c;Qt的ui文件在编译时&#xff0c;会自动生成一个ui_xxxxx.h的头文件&#xff0c;打开一看&#xff0c;其实就是将摆放的控件new出来以及布局的代码。 只要用Qt提供的uic.exe工具&#xff0c;自己也可以将ui文件输出为代码文件…

c# 笔记 winform添加右键菜单,获取文件大小 ,多条件排序OrderBy、ThenBy,list<double>截取前5个

Winform右键菜单‌ 要在C# Winform应用程序中添加右键菜单&#xff0c;‌你可以按照以下步骤操作&#xff1a;‌ 1.‌创建菜单项‌ 在Form的构造函数或加载事件中&#xff0c;‌创建ContextMenuStrip控件的实例&#xff0c;‌并为其添加菜单项。‌ 2.‌绑定到控件‌ 将Con…

c++ websocket简单讲解

只做简单讲解。 一.定义和原理 WebSocket 是从 HTML5 开始⽀持的⼀种⽹⻚端和服务端保持⻓连接的消息推送机制&#xff0c;传统的 web 程序都是属于 "⼀问⼀答" 的形式&#xff0c;即客⼾端给服务器发送了⼀个 HTTP 请求&#xff0c;服务器给客⼾端返回⼀个 HTTP 响…

Java 入门指南:Java 并发编程 —— 并发容器 PriorityBlockingQueue

BlockingQueue BlockingQueue 是Java并发包&#xff08;java.util.concurrent&#xff09;中提供的一个阻塞队列接口&#xff0c;它继承自 Queue 接口。 BlockingQueue 中的元素采用 FIFO 的原则&#xff0c;支持多线程环境并发访问&#xff0c;提供了阻塞读取和写入的操作&a…

视频汇聚平台LntonAIServer视频质量诊断功能--偏色检测与噪声检测

随着视频监控技术的不断进步&#xff0c;视频质量成为了决定监控系统性能的关键因素之一。LntonAIServer新增的视频质量诊断功能&#xff0c;特别是偏色检测和噪声检测&#xff0c;进一步强化了视频监控系统的可靠性和实用性。下面我们将详细介绍这两项功能的技术细节、应用场景…

【AI】Pytorch_损失函数优化器

建议点赞收藏关注&#xff01;持续更新至pytorch大部分内容更完。 本文已达到10w字&#xff0c;故按模块拆开&#xff0c;详见目录导航。 整体框架如下 数据及预处理 模型及其构建 损失函数及优化器 本节目录 损失函数创建损失函数 &#xff08;共18个&#xff09;nn.CrossEnt…

【多场景应用】基于杰发科技AC7840x的Mini LED背光驱动设计

应用场景&#xff1a; 在汽车应用中&#xff0c;Mini LED背光驱动设计主要用于仪表盘、中控屏和车载娱乐系统等显示屏。这项技术可以显著提升显示效果&#xff0c;提供更高的亮度、更深的黑色和更广的色域&#xff0c;使得图像更加生动逼真&#xff0c;尤其在强光和宽温度范围…

Redis 篇-深入了解查询缓存与缓存所带来的问题(读写不一致、缓存穿透、缓存雪崩、缓存击穿)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 本章目录 1.0 什么是缓存 2.0 项目中具体如何添加缓存 3.0 添加缓存后所带来的问题 3.1 读写不一致问题 3.1.1 缓存更新策略 3.1.2 具体实现缓存与数据库的双写一致 3.2 缓存穿…

【日记】想见珍一面怎么就这么难(985 字)

正文 想见珍一面怎么就这么难…… 事故频发。昨天说考试时间跟机票时间冲突了&#xff0c;最后结果出来了&#xff0c;改签了&#xff0c;并且差价不补。我不干&#xff0c;他们也不干。因为上级行给我们行长施压&#xff0c;于是我们行长给我施压。最后要到了国庆之前拔智齿的…

华为 HCIP-Datacom H12-821 题库 (6)

有需要题库的可以看主页置顶 V群仅进行学习交流 1.转发表中 FLAG 字段中B 的含义是&#xff1f; A、可用路由 B、静态路由 C、黑洞路由 D、网关路由 答案&#xff1a;C 解析&#xff1a; 可用路由用U 表示&#xff0c;静态路由用 S 表示&#xff0c;黑洞路由用 B 表示&#x…

笔试,牛客.kotori和n皇后​,牛客.AOE还是单体

目录 牛客.kotori和n皇后​编辑 牛客.AOE还是单体 牛客.kotori和n皇后 想起来&#xff0c;我之前还写过n皇后的题&#xff0c;但是这个我开始只能想到暴力解法 判断是不是斜对角线&#xff0c;联想yxb和y-xb,假如在一条线上&#xff0c;那么他们的x和y会对应成比例&#xff0c…

【弱监督时间动作定位】Probabilistic Vision-Language Representation for WSTAL 论文阅读

Probabilistic Vision-Language Representation for Weakly Supervised Temporal Action Localization 论文阅读 Abstract1 Introduction2 RELATEDWORK2.1 Weakly Supervised Temporal Action Localization2.2 Vision Language Pre-training2.3 Probabilistic Representation 3…

RocketMQ高级特性四-消息过滤

目录 前言 Broker端过滤 定义与概述 消息过滤分类 原理机制 使用场景 优缺点 Java代码示例 - Tag过滤 Java代码示例 - SQL92过滤 客户端过滤 定义与概述 原理机制 使用场景 优缺点 Java代码示例 总结 前言 消息过滤是RocketMQ的一项高级特性&#xff0c;它允许…

常见HTTP状态码、APUD响应状态字及含义

目录 一、HTTP状态码 二、APDU指令码 一、HTTP状态码 HTTP状态&#xff08;HTTP Status Code&#xff09;是用以表示网页服务器超文本传输协议响应状态的3位数字代码。 关于HTTP状态码更加详细介绍推荐阅读&#xff1a; http://t.csdnimg.cn/qSJv6http://t.csdnimg.cn/qSJv…

光敏电阻传感器详解(STM32)

目录 一、介绍 二、传感器原理 1.光敏电阻传感器介绍 2.原理图 三、程序设计 main.c文件 ldr.h文件 ldr.c文件 四、实验效果 五、资料获取 项目分享 一、介绍 光敏电阻器是利用半导体的光电导效应制成的一种电阻值随入射光的强弱而改变的电阻器&#xff0c;又称为光…

基于树莓派的儿童音频播发器—Yoto

Raspberry Pi 的开发可能性使吸引人的、以儿童为中心的音频播放器得以成型 Yoto Player 为孩子们提供了拥有和控制的绝佳体验&#xff0c;同时不会增加屏幕时间。得益于 Raspberry Pi 以及我们认可的经销商提供的支持和专业知识&#xff0c;Yoto Player 在英国取得了成功。 Yo…

七款最佳的渗透测试工具(非常详细)零基础入门到精通,收藏这一篇就够了

渗透测试工具是模拟对计算机系统、网络或 Web 应用程序的网络攻击的软件应用程序&#xff0c;它们的作用是在实际攻击者之前发现安全漏洞。它们可以作为系统的压力测试&#xff0c;揭示哪些区域可能会受到真正的威胁。 本文我将介绍七款最佳的渗透测试工具。 1 Kali Linux K…

Maven入门:自动化构建工具的基本概念与配置

一、什么是Maven 目前无论使用IDEA还是Eclipse等其他IDE&#xff0c;使用里面 ANT 工具帮助我们进行编译&#xff0c;打包运行等工作。Apache基于ANT进行了升级&#xff0c;研发出了全新的自动化构建工具Maven。 Maven使用项目对象模型&#xff08;POM-Project Object Model&…

视频合并在线工具哪个好?好用的视频合并工具推荐

当我们手握一堆零散却各有千秋的视频片段时&#xff0c;是否曾幻想过它们能像魔法般合并成一部完整、流畅的故事&#xff1f; 别担心&#xff0c;今天咱们就来一场“视频合并大冒险”&#xff0c;揭秘几款视频合并软件手机免费工具&#xff0c;帮助你在指尖上实现创意无限的视…

四、配置三层交换实验组网

一、实验拓扑 二、实验目的 通过配置交换机&#xff0c;令不同vlan间的主机能够互相通信 三、实验步骤 SW12 <Huawei>undo terminal monitor Info: Current terminal monitor is off. <Huawei>system-view Enter system view, return user view with CtrlZ. [H…