• 从零开始的C++(十二)


    长风破浪会有时,直挂云帆济沧海。

    List:

    链式结构,类似带头双向循环链表

    常见成员函数:

    push_back():尾插      pop_back():尾删    push_front():头插     pop_front():头删

    特征:开辟的空间不连续,因此插入、删除结点效率高。但不支持随机访问,无法指定下标来快速访问。

    list模拟实现:

    list模拟实现包含三个类,一部分是表示链表的每个结点的类、一部分是迭代器、一部分是链表的头结点(也是哨兵结点,特征是指向第一个有效结点)。

    链表结点:

    1. template<class T>
    2. struct ListNode
    3. {
    4. ListNode(const T& val = T()) : _pPre(nullptr), _pNext(nullptr), _val(val)
    5. {}
    6. ListNode<T>* _pPre;
    7. ListNode<T>* _pNext;
    8. T _val;
    9. };

    由于是双向循环链表,所以必须有前后两个指针同时_val用于保存值。

    迭代器:

    1. template<class T, class Ref, class Ptr>
    2. class ListIterator
    3. {
    4. typedef ListNode<T>* PNode;
    5. typedef ListIterator<T, Ref, Ptr> Self;
    6. public:
    7. ListIterator(PNode pNode = nullptr) :_pNode(pNode)
    8. {}
    9. ListIterator(const Self& l) : _pNode(l._pNode)
    10. {}
    11. Ref operator*()
    12. {
    13. return _pNode->_val;
    14. }
    15. Ptr operator->()
    16. {
    17. return &*this;
    18. }
    19. Self& operator++()
    20. {
    21. _pNode = _pNode->_pNext;
    22. return *this;
    23. }
    24. Self operator++(int)
    25. {
    26. Self temp(*this);
    27. _pNode = _pNode->_pNext;
    28. return temp;
    29. }
    30. Self& operator--()
    31. {
    32. _pNode = _pNode->_pPre;
    33. return *this;
    34. }
    35. Self& operator--(int)
    36. {
    37. Self temp(*this);
    38. _pNode = _pNode->_pPre;
    39. return temp;
    40. }
    41. bool operator!=(const Self& l)
    42. {
    43. return _pNode != l._pNode;
    44. }
    45. bool operator==(const Self& l)
    46. {
    47. return !(*this != l);
    48. }
    49. public:
    50. PNode _pNode;
    51. };

    首先,迭代器的功能类似与指针,因此唯一的元素是结点指针类型。

    迭代器需要实现遍历,因此需要有++、--来进行迭代器的移动,有!=、==来判断是否到达终止条件,有*、->来实现解引用。此处迭代器适配const类型和非const类型,而const与非const的区别在于解引用的返回类型是否有const修饰,此处返回类型用Ref和Ptr代替,Ref会根据数据判断是T&还是const T&,Ptr同理。

    表头节点:

    1. template<class T>
    2. class list
    3. {
    4. typedef ListNode<T> Node;
    5. typedef Node* PNode;
    6. public:
    7. typedef ListIterator<T, T&, T*> iterator;
    8. typedef ListIterator<T, const T&, const T&> const_iterator;
    9. public:
    10. ///
    11. // List的构造
    12. list()
    13. {
    14. CreateHead();
    15. }
    16. list(int n, const T& value = T())
    17. {
    18. CreateHead();
    19. for (int i = 0; i < n; ++i)
    20. push_back(value);
    21. }
    22. template <class Iterator>
    23. list(Iterator first, Iterator last)
    24. {
    25. CreateHead();
    26. while (first != last)
    27. {
    28. push_back(*first);
    29. ++first;
    30. }
    31. }
    32. list(const list<T>& l)
    33. {
    34. CreateHead();
    35. // 用l中的元素构造临时的temp,然后与当前对象交换
    36. list<T> temp(l.cbegin(), l.cend());
    37. this->swap(temp);
    38. }
    39. list<T>& operator=(const list<T> l)
    40. {
    41. this->swap(l);
    42. return *this;
    43. }
    44. ~list()
    45. {
    46. clear();
    47. delete _pHead;
    48. _pHead = nullptr;
    49. }
    50. ///
    51. // List Iterator
    52. iterator begin()
    53. {
    54. return iterator(_pHead->_pNext);
    55. }
    56. iterator end()
    57. {
    58. return iterator(_pHead);
    59. }
    60. const_iterator begin()const
    61. {
    62. return const_iterator(_pHead->_pNext);
    63. }
    64. const_iterator end()const
    65. {
    66. return const_iterator(_pHead);
    67. }
    68. ///
    69. // List Capacity
    70. size_t size()const
    71. {
    72. size_t size = 0;
    73. ListNode* p = _pHead->_pNext;
    74. while (p != _pHead)
    75. {
    76. size++;
    77. p = p->_pNext;
    78. }
    79. return size;
    80. }
    81. bool empty()const
    82. {
    83. return size() == 0;
    84. }
    85. // List Access
    86. T& front()
    87. {
    88. assert(!empty());
    89. return _pHead->_pNext->_val;
    90. }
    91. const T& front()const
    92. {
    93. assert(!empty());
    94. return _pHead->_pNext->_val;
    95. }
    96. T& back()
    97. {
    98. assert(!empty());
    99. return _pHead->_pPre->_val;
    100. }
    101. const T& back()const
    102. {
    103. assert(!empty());
    104. return _pHead->_pPre->_val;
    105. }
    106. // List Modify
    107. void push_back(const T& val)
    108. {
    109. insert(end(), val);
    110. }
    111. void pop_back()
    112. {
    113. erase(--end());
    114. }
    115. void push_front(const T& val)
    116. {
    117. insert(begin(), val);
    118. }
    119. void pop_front()
    120. {
    121. erase(begin());
    122. }
    123. // 在pos位置前插入值为val的节点
    124. iterator insert(iterator pos, const T& val)
    125. {
    126. PNode pNewNode = new Node(val);
    127. PNode pCur = pos._pNode;
    128. // 先将新节点插入
    129. pNewNode->_pPre = pCur->_pPre;
    130. pNewNode->_pNext = pCur;
    131. pNewNode->_pPre->_pNext = pNewNode;
    132. pCur->_pPre = pNewNode;
    133. return iterator(pNewNode);
    134. }
    135. // 删除pos位置的节点,返回该节点的下一个位置
    136. iterator erase(iterator pos)
    137. {
    138. // 找到待删除的节点
    139. PNode pDel = pos._pNode;
    140. PNode pRet = pDel->_pNext;
    141. // 将该节点从链表中拆下来并删除
    142. pDel->_pPre->_pNext = pDel->_pNext;
    143. pDel->_pNext->_pPre = pDel->_pPre;
    144. delete pDel;
    145. return iterator(pRet);
    146. }
    147. void clear()
    148. {
    149. iterator p = begin();
    150. while (p != end())
    151. {
    152. p = erase(p);
    153. }
    154. _pHead->_pPre = _pHead;
    155. _pHead->_pNext = _pHead;
    156. }
    157. void swap(list<T>& l)
    158. {
    159. PNode tmp = _pHead;
    160. _pHead = l._pHead;
    161. l._pHead = tmp;
    162. }
    163. private:
    164. void CreateHead()
    165. {
    166. _pHead = new Node;
    167. _pHead->_pPre = _pHead;
    168. _pHead->_pNext = _pHead;
    169. }
    170. PNode _pHead;
    171. };
    172. }

    对于默认构造函数,此处调用CreateHead函数,实现了创建一个Node类型的结点,同时结点的两个指针都指向自己。

    对于传值构造函数,首先调用CreateHead函数,实现了创建一个Node类型的结点,然后复用了push_back函数,实现结点的插入。

    对于传迭代器的构造,先创建头结点,然后利用迭代器遍历,实现数字插入。

    对于拷贝构造,先创建一个头结点,然后复用了迭代器的构造方式产生一个临时对象,然后交换临时对象和当前要创建的对象的成员变量,实现拷贝,同时利用临时对象销毁创建的头结点。此处如果不想开头创建一个头结点,则析构函数必须更改一下,只有头结点不为空的时候才调用clear函数,否则clear函数中调用的begin函数会出现非法访问的情况。

    对于赋值函数,原理和拷贝构造一致,都是利用临时对象来替换,实现赋值操作。

    对于头插、头删、尾插、尾删,复用insert和erase函数,但需要注意传入的迭代器是谁。

    insert:在迭代器所指位置之前插入一个结点,同时返回插入结点的迭代器。

    erase:删除迭代器所指位置的结点,返回下一个结点的迭代器。

    clear:利用迭代器遍历链表,利用erase删除链表结点,实现删除链表除了表头节点之外的所有结点。

    析构函数:在clear函数删除除表头节点外的所有结点后,销毁表头节点。

  • 相关阅读:
    NTT简介
    docker部署ES及kibana整个流程
    PCL 判断点位于球内外侧
    光伏储能MPPT控制系统如何进行浪涌静电保护?
    Golang开发--计时器(Timer)和定时器(Ticker)
    C++设计模式之模板方法模式
    tcpdump进行IP抓包
    FFmpeg视音频分离器----向雷神学习
    Spring的IOC原理
    【Pytorch深度学习开发实践学习】【AlexNet】经典算法复现-Pytorch实现AlexNet神经网络(1)model.py
  • 原文地址:https://blog.csdn.net/yyssas/article/details/134058157