C++:浅谈双向迭代器--以list迭代器为例

前言

这篇文章实际上我感觉还是没有讲清楚,有问题的话,还请读者在评论区指出。

迭代器主要是分为随机迭代器和双向迭代器。

随机迭代器由于在物理空间上是连续的,所以支持++,--,+,-,>,<,+=,-=等操作。

双向迭代器在物理空间上不连续,不能够支持+,-,>,<,等操作,但支持++,--。

这篇文章主要讲解双向迭代器如何在物理空间不连续的情况下实现迭代器的通用功能。

本文以list容器为例。

这篇文章实际上我感觉还是没有讲清楚,有问题的话,还请读者在评论区指出。

正文

我们知道list容器的每一个节点都是有三个元素,prev和next指针以及数据data。

这三个元素封装到Node类里面,作为list的节点。

首先,假设我们已经有了auto it = begin()迭代器,我们想要访问下一个元素怎么办,很显然,需要it = it->next,才能访问下一个元素。

那么矛盾来了,迭代器毕竟不是指针啊,不能这么使用,迭代器肯定是使用++it来访问下一个元素,诶,怎么解决这个问题呢?

是不是可以想到运算符重载来改变对象的行为。

但是,还是存在一些问题的,如果在list类里面去重载这些迭代器所需要的运算符,当其他的函数需要使用这些运算符本身时,那不就又出现矛盾了嘛。

怎么解决呢?

当...当...当...
我们可以考虑使用**封装的设计模式,将list的原生指针和这些需要重载的运算符重载到一个类里面**,这样就可以独立使用,而且不会干扰list的其他函数。

模拟实现中遇到的困难和解决办法

a、要时刻清楚每一个类是干嘛的,list里面类是比较多的,很容以把自己绕糊涂

b、要学会代码复用,比如写完了insert,那push_back直接调用insert函数就行

c、一处比较有意思的设计技巧
在设计iterator和const_iterator的时候,怎样快速的实现两个类呢?要不要傻乎乎的把类似的代码写两遍呢?
哎呀,我们是学过模版的嘛,重复的工作丢给编译器去实现就好啦。
template<class T,class Ref,class Ptr>
我们给迭代器类里面放三个模版参数,iterator实例化的时候就传T&和T就可以了,const_iterator实例化的时候传const T&和const T ,编译器就会帮我们实现这两个类。
如果只传一个模版参数的话,那就得傻乎乎的写两个类了。

ps:实际上传三个模版参数也是实现两个类,只是编译器帮我们把重复的工作实现了而已。

代码

C++ 复制代码
template<class T,class Ref,class Ptr>
struct ListIterator
{
	typedef ListNode<T> Node;
	typedef ListIterator<T, Ref, Ptr> Self;

	Node* _node;

	ListIterator(Node* node):_node(node) {}

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

	Self operator++(int)
	{
		Self ret(_node);
		_node = _node->_next;

		return ret;
	}

	Self& operator--()
	{
		_node = _node->_prev;
		return *this;
	}

	Self operator--(int)
	{
		Self ret(_node);
		_node = _node->_prev;
		return ret;
	}

	Ref operator*()//解引用
	{
		return _node->_data;
	}

	Ptr operator->()//成员访问操作符的重载
	{
		return &(_node->_data);
	}
	
	bool operator!=(const Self& it)
	{
		return _node != it._node;
	}

	bool operator==(const Self& it)
	{
		return _node == it._node;
	}
};
相关推荐
weixin_4373982118 分钟前
转Go学习笔记
linux·服务器·开发语言·后端·架构·golang
津津有味道27 分钟前
Qt C++串口SerialPort通讯发送指令读写NFC M1卡
linux·c++·qt·串口通信·serial·m1·nfc
StrongerIrene28 分钟前
rs build 的process.env的值undefined解决方案
开发语言·javascript·ecmascript
风逸hhh40 分钟前
python打卡day58@浙大疏锦行
开发语言·python
Q_970956391 小时前
java+vue+SpringBoo足球社区管理系统(程序+数据库+报告+部署教程+答辩指导)
java·开发语言·数据库
傅里叶的耶1 小时前
C++系列(二):告别低效循环!选择、循环、跳转原理与优化实战全解析
c++·visual studio
Vitta_U1 小时前
MFC的List Control自适应主界面大小
c++·list·mfc
为了更好的明天而战1 小时前
Java 中的 ArrayList 和 LinkedList 区别详解(源码级理解)
java·开发语言
JosieBook2 小时前
【Java编程动手学】Java中的数组与集合
java·开发语言·python
qq_589568102 小时前
element-plus按需自动导入的配置 以及icon图标不显示的问题解决
开发语言·javascript·ecmascript