当前位置:首页 » 《资源分享》 » 正文

[ C++ ] STL---反向迭代器的模拟实现

6 人参与  2024年03月28日 09:15  分类 : 《资源分享》  评论

点击全文阅读


目录

前言:

反向迭代器简介

list反向迭代器的模拟实现

 反向迭代器的模拟实现(适配器模式)

SGI版本STL反向迭代器源码

STL库中解引用操作与出口设计

适配list的反向迭代器

适配vector的反向迭代器


前言:

反向迭代器是一种特殊类型的迭代器,它可以逆向遍历容器中的元素,与正向迭代器相比,反向迭代器从容器的末尾开始,向前遍历到容器的起始位置,反向迭代器提供了一种方便的方式来反向访问容器中的元素,特别适用于需要逆序处理数据的场景;

在C++标准库中,反向迭代器通常通过rbegin()和rend()成员函数来获取,一般情况下,rbegin()返回指向容器最后一个元素的迭代器,而rend()返回指向容器起始位置前一个元素的迭器;

反向迭代器简介

#include <iostream>#include <list>using namespace std;int main(){list<int> lt;lt.push_back(10);lt.push_back(20);lt.push_back(30);lt.push_back(40);lt.push_back(50);//正向遍历list<int>::iterator it = lt.begin();while (it != lt.end()){cout << *it << " ";++it;}cout << endl;//反向遍历list<int>::reverse_iterator rit = lt.rbegin();while (rit != lt.rend()){cout << *rit << " ";++rit;}cout << endl;return 0;}

运行结果:

list反向迭代器的模拟实现

思考一: 

从行为方式上,反向迭代器与正向迭代器的区别是什么?

正向迭代器与反向迭代器的除了++、- -等接口外毫无区别;

若只实现链表的反向迭代器,如何实现?

链表普通迭代器实现入口:[ C++ ] STL---list的模拟实现-CSDN博客

链表反向迭代器实现步骤:

拷贝普通迭代器,类名修改为__List_reverse_iterator;修改前置++、后置++、前置--、后置-- 接口;list类中重定义reverse_iterator与const_reverse_iterator;
//list反向迭代器实现代码template<class T, class Ref, class Ptr>struct __List_reverse_iterator//修改类名{typedef ListNode<T> Node;Node* _node;__List_reverse_iterator(Node* node):_node(node){}typedef __List_reverse_iterator<T> self;//修改++/--接口的指向self& operator++(){_node = _node->_prev;return *this;}//it++self operator++(int){self tmp(*this);_node = _node->_prev;return tmp;}//--itself& operator--(){_node = _node->_next;return *this;}//it--self operator--(int){self tmp(*this);_node = _node->_next;return tmp;}bool operator==(const self& s){return _node == s._node;}bool operator!=(const self& s){return _node != s._node;}Ref operator*(){return _node->_data;}Ptr operator->(){return &_node->_data;}};template<class T>class list{typedef ListNode<T> Node;public:    //list类中重定义typedef __List_reverse_iterator<T, T&, T*> reverse_iterator;//重命名反向迭代器typedef __List_reverse_iterator<T, const T&, const T*> const_reverse_iterator;//重命名const反向迭代器//......private:Node* _head;};

 反向迭代器的模拟实现(适配器模式)

上述实现list反向迭代器的方式会在同一份文件中,存在普通迭代器与反向迭代器两个类并且两个类中仅有个别接口略有差异,代码冗余,导致可读性变差,更好的实现方案是什么?

解决方案:

vector/list/二叉树等容器均要面临反向迭代器的实现问题,如此便采用适配器模式,即链表的正向迭代器适配(改造)出链表的反向迭代器,vector的正向迭代器适配(改造)出vector的反向迭代器;

SGI版本STL反向迭代器源码

//stl_iterator.h文件template <class _Iterator>class reverse_iterator {protected:  _Iterator current;public:  typedef typename iterator_traits<_Iterator>::iterator_category          iterator_category;  typedef typename iterator_traits<_Iterator>::value_type          value_type;  typedef typename iterator_traits<_Iterator>::difference_type          difference_type;  typedef typename iterator_traits<_Iterator>::pointer          pointer;  typedef typename iterator_traits<_Iterator>::reference          reference;  typedef _Iterator iterator_type;  typedef reverse_iterator<_Iterator> _Self;public:  reverse_iterator() {}  explicit reverse_iterator(iterator_type __x) : current(__x) {}  reverse_iterator(const _Self& __x) : current(__x.current) {}#ifdef __STL_MEMBER_TEMPLATES  template <class _Iter>  reverse_iterator(const reverse_iterator<_Iter>& __x)    : current(__x.base()) {}#endif /* __STL_MEMBER_TEMPLATES */      iterator_type base() const { return current; }  reference operator*() const {    _Iterator __tmp = current;    return *--__tmp;  }#ifndef __SGI_STL_NO_ARROW_OPERATOR  pointer operator->() const { return &(operator*()); }#endif /* __SGI_STL_NO_ARROW_OPERATOR */  _Self& operator++() {    --current;    return *this;  }  _Self operator++(int) {    _Self __tmp = *this;    --current;    return __tmp;  }  _Self& operator--() {    ++current;    return *this;  }  _Self operator--(int) {    _Self __tmp = *this;    ++current;    return __tmp;  }  _Self operator+(difference_type __n) const {    return _Self(current - __n);  }  _Self& operator+=(difference_type __n) {    current -= __n;    return *this;  }  _Self operator-(difference_type __n) const {    return _Self(current + __n);  }  _Self& operator-=(difference_type __n) {    current += __n;    return *this;  }  reference operator[](difference_type __n) const { return *(*this + __n); }  }; 
//stl_list.h文件reverse_iterator rbegin() { return reverse_iterator(end()); }const_reverse_iterator rbegin() const { return const_reverse_iterator(end()); }reverse_iterator rend(){ return reverse_iterator(begin()); }const_reverse_iterator rend() const{ return const_reverse_iterator(begin()); }

STL库中解引用操作与出口设计

STL库中设计了一种对称结构,正向迭代器的开始位置为反向迭代器的结束位置,反向迭代器的开始位置即正向迭代器的结束位置;

适配list的反向迭代器

实现思路:

将正向迭代器作为模板参数传递给反向迭代器的成员变量cur,如此反向迭代器就可以自动匹配容器,反向迭代器就可以统一实现复用了:

//ReverseIterator.h文件//第一个模版参数为正向迭代器,利用正向迭代器改造反向迭代器;//第二个模版参数Ref定义反向迭代器的数据类型,数据类型分为const类型/非const类型;//第三个模版参数Ptr定义自定义类型指针所指向的数据的的读写属性;template<class Iterator,class Ref,class Ptr>struct ReverseIterator{//成员变量cur,cur为正向迭代器类所定义的对象Iterator cur;//ReverseIterator<Iterator,Ref,Ptr> operator++()typedef ReverseIterator<Iterator, Ref, Ptr> Self;Self& operator++(){--cur;//调用cur对象所属类的operator--()函数return *this;}Self operator++(int){Self tmp(cur);--cur;return tmp;}Self& operator--(){++cur;//调用cur对象所属类的operator++()函数return *this;}Self operator--(int){Self tmp(cur);++cur;return tmp;}// !=/== 本质比较正向迭代器也即结点的指针bool operator!=(const Self& s){return cur != s.cur;}bool operator==(const Self& s){return cur == s.cur;}//构造函数(正向迭代器构造反向迭代器)ReverseIterator(Iterator it):cur(it){}Ref operator*(){Iterator tmp = cur;--tmp;return *tmp;}//需要对象指针即对象地址,所以先解引用再取&Ptr operator->(){return &(operator*());}};
list类中统一名称并且需要提供出口;
//list.h文件代码节选template<class T>class list{typedef ListNode<T> Node;public:typedef __List_iterator<T, T&, T*> iterator;typedef __List_iterator<T, const T&, const T*> const_iterator;//list类中统一名称typedef  ReverseIterator<iterator, T&, T*> reverse_iterator;typedef  ReverseIterator<iterator, const T&, const T*> const_reverse_iterator;//搭配出口,list类中实现rbegin()、rend()reverse_iterator rbegin(){return reverse_iterator(end());}reverse_iterator rend(){return reverse_iterator(begin());}//......}

适配vector的反向迭代器

//vector.h文件代码节选template<class T>class vector{public:typedef T* iterator;typedef const T* const_iterator;//vector中统一名称typedef  ReverseIterator<iterator, T&, T*> reverse_iterator;typedef  ReverseIterator<iterator, const T&, const T*> const_reverse_iterator;//搭配出口,vector中实现rbegin()、rend()reverse_iterator rbegin(){return reverse_iterator(end());}reverse_iterator rend(){return reverse_iterator(begin());}//......private:iterator _start = nullptr;iterator _finish = nullptr;iterator _endofstorage = nullptr;};

欢迎大家批评指正,博主会持续输出优质内容,谢谢大家观看,码字画图不易,希望大家给个一键三连支持~ 你的支持是我创作的不竭动力~


点击全文阅读


本文链接:http://m.zhangshiyu.com/post/86910.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

最新文章

  • 林晚夏江肆年(进错房,嫁给八零最牛特种兵在线阅读)全文免费阅读无弹窗大结局_(林晚夏江肆年)进错房,嫁给八零最牛特种兵在线阅读免费阅读全文最新章节列表_笔趣阁(林晚夏江肆年) -
  • 进错房,嫁给八零最牛特种兵完整版阅读小说(林晚夏江肆年)全文免费阅读无弹窗大结局_(进错房,嫁给八零最牛特种兵完整版阅读)林晚夏江肆年免费阅读全文最新章节列表_笔趣阁(进错房,嫁给八零最牛特种兵完整版阅读) -
  • 新雪藏旧事全文全文(商云萝周砚京)全文免费阅读无弹窗大结局_(新雪藏旧事全文小说免费阅读)最新章节列表_笔趣阁(新雪藏旧事全文) -
  • 在线免费小说重生七零替嫁:不嫁教授,嫁军官_乔珊珊乔婉月新热门小说_热门小说乔珊珊乔婉月
  • 免费小说《冯云漪厉晋泽》已完结(冯云漪厉晋泽)热门小说大结局全文阅读笔趣阁
  • 祁兰湘邵黎晖小说_祁兰湘邵黎晖完整版大结局小说免费阅读
  • 完整免费小说老公心疼青梅将她留宿新房,却将怀孕的我赶出家门(乔玥傅慎行姜禾)_老公心疼青梅将她留宿新房,却将怀孕的我赶出家门(乔玥傅慎行姜禾)完本小说免费阅读(乔玥傅慎行姜禾)
  • 新雪藏旧事:结局+番外+完结免费小说在线阅读_小说完结推荐新雪藏旧事:结局+番外+完结商云萝周砚京热门小说
  • 初逢青山梦长安(顾怀瑾沈书妤)阅读 -
  • 无删减版《绝对权力:从天崩开局走上官途巅峰》在线免费阅读
  • 《绝对权力:从天崩开局走上官途巅峰》小说在线试读,《绝对权力:从天崩开局走上官途巅峰》最新章节目录
  • 裴泽苏星辰何娇(满目星辰不及你小说)精彩章节在线阅读

    关于我们 | 我要投稿 | 免责申明

    Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1