C++STL模板之——list(简化源码,模拟源码)

简介: C++STL模板之——list(简化源码,模拟源码)

众所周知,C++给我们底层搬砖人提供了很多便捷的数据结构,让我们能偶尔偷懒,list就是其中之一,现在让我们来了解一下它吧



一,原理

1)底层大致结构

list底层是由带头双向链表构成的,带头即带哨兵位,双向就是可以从前往后遍历也可从后往前遍历。那这个时候就有人好奇哨兵位指向前一个结点的指针指向哪里?最后一个结点的next指针指向哪里?答案是哨兵位的prv指针指向最后一个结点,最后一个结点的next则指向哨兵位。

2)迭代器

大家有没有发现一个问题,底层是链表,链表的地址是随机的,那迭代器的++不是有问题吗?这是当然的,所有我们需要单独处理迭代器,无法像vector一样的直接++就可拿到下一个元素的地址,那我们应该怎么解决呢?答案是将迭代器单独写一个类,对++,--,等函数进行重载,封装之后就可利用链表里面的指针拿到下一个或者上一个元素的地址,同时begin()和end()返回的就应该是这个迭代器类。

3)模板

因为要适配各种元素,所有这里必须使用模板。而模板的加入必然会让底层的难度和结构更加复杂,同时我们要考虑迭代器类也要加模板以此来适配各种类。

二,模拟源码

1)链表结点

我们要分析链表里面的结构,我们要从需求出发,里面需要什么,首先两个链表指针分别指前和指后,还有应该模板元素用于存储使用者的值,构造函数用于调用初始化,而析构函数建议交给list类。

template<class T>
  struct ListNode
  {
    ListNode(const T& val = T()): _pPre(nullptr), _pNext(nullptr), _val(val)
    {}
    ListNode<T>* _pPre;//指向前一个结点
    ListNode<T>* _pNext;//指向后一个结点
    T _val;             //真正存储的有效值
  };
2)list类部分

首先我们要考虑,list的私有成员是什么?那肯定是结点啊,从前面的描述可知,list底层结构是带头双向链表,因此私有成员应该是哨兵位指针,由于我们会使用Node* 很多次,因此我们命名为PNode,增加可读性。这里面唯一值得我们注意的就是end()和begin()函数,因为链表底层是不连续的地址,因此我们要将迭代器及其操作封装,begin和end就应该返回被包装的类。还有麻烦的一点就是有些函数需要提供const版本,防止调用的时候找不到匹配的函数。其余的可以按照数据结构链表的基础来写,如果有不懂的可以看我以往的数据结构初阶博客。

//list类
  template<class T>
  class list
  {
    typedef ListNode<T> Node;
    typedef Node* PNode;
  public:
    typedef ListIterator<T, T&, T*> iterator;
    typedef ListIterator<T, const T&, const T&> const_iterator;
  public:
    ///
    // List的构造
    list()
    {
      CreateHead();
    }
    list(int n, const T& value = T())
    {
      CreateHead();
      for (int i = 0; i < n; ++i)
        push_back(value);
    }
    template <class Iterator>
    list(Iterator first, Iterator last)
    {
      CreateHead();
      while (first != last)
      {
        push_back(*first);
        ++first;
      }
    }
    list(const list<T>& l)
    {
      CreateHead();
      // 用l中的元素构造临时的temp,然后与当前对象交换
      list<T> temp(l.cbegin(), l.cend());
      this->swap(temp);
    }
    list<T>& operator=(const list<T> l)
    {
      this->swap(l);
      return *this;
    }
    ~list()
    {
      clear();
      delete _pHead;
      _pHead = nullptr;
    }
    ///
    // List Iterator
    iterator begin()
    {
      return iterator(_pHead->_pNext);
    }
    iterator end()
    {
      return iterator(_pHead);
    }
    const_iterator begin()const
    {
      return const_iterator(_pHead->_pNext);
    }
    const_iterator end()const
    {
      return const_iterator(_pHead);
    }
    ///
    // List Capacity
    size_t size()const
    {
      size_t size = 0;
      ListNode *p = _pHead->_pNext;
      while(p != _pHead)
      {
        size++;
        p = p->_pNext;
      }
      return size;       
    }
    bool empty()const
    {
      return size() == 0;
    }
    
    // List Access
    T& front()
    {
      assert(!empty());
      return _pHead->_pNext->_val;
    }
    const T& front()const
    {
      assert(!empty());
      return _pHead->_pNext->_val;
    }
    T& back()
    {
      assert(!empty());
      return _pHead->_pPre->_val;
    }
    const T& back()const
    {
      assert(!empty());
      return _pHead->_pPre->_val;
    }
    
    // List Modify
    void push_back(const T& val)
    {
      insert(end(), val);
    }
    void pop_back()
    {
      erase(--end());
    }
    void push_front(const T& val)
    {
      insert(begin(), val);
    }
    void pop_front()
    {
      erase(begin());
    }
    // 在pos位置前插入值为val的节点
    iterator insert(iterator pos, const T& val)
    {
      PNode pNewNode = new Node(val);
      PNode pCur = pos._pNode;
      // 先将新节点插入
      pNewNode->_pPre = pCur->_pPre;
      pNewNode->_pNext = pCur;
      pNewNode->_pPre->_pNext = pNewNode;
      pCur->_pPre = pNewNode;
      return iterator(pNewNode);
    }
    // 删除pos位置的节点,返回该节点的下一个位置
    iterator erase(iterator pos)
    {
      // 找到待删除的节点
      PNode pDel = pos._pNode;
      PNode pRet = pDel->_pNext;
      // 将该节点从链表中拆下来并删除
      pDel->_pPre->_pNext = pDel->_pNext;
      pDel->_pNext->_pPre = pDel->_pPre;
      delete pDel;
      return iterator(pRet);
    }
    void clear()
    {
      iterator p = begin();
      while(p != end())
      {
        p = erase(p);
      }
           
           _pHead->_pPrev = _pHead;
           _pHead->_pNext = _pHead;
    }
    void swap(List<T>& l)
    {
      pNode tmp = _pHead;
      _pHead = l._pHead;
      l._pHead = tmp;
    }
  private:
    void CreateHead()
    {
      _pHead = new Node;
      _pHead->_pPre = _pHead;
      _pHead->_pNext = _pHead;
    }
    PNode _pHead;
  };
}
3)迭代器类

当然,我们首先考虑的还是私有成员,私有成员应该是链表结点,结点不一定是哨兵位,可能是任何一个结点,因为迭代器可以指向任何一个元素,然后就要对迭代器的操作符重写,手动进行地址的加减或者解引用,其中值得人注意的是->操作符重写,很多人可能好奇为什么返回的是结点地址,因为其实a->其实是等价于&a->->,我们返回这个地址便于我们拿到私有成员进行操作。最难理解的一个点是为什么要提供一个ListIterator<T, Ref, Ptr> 的构造函数,因为ListIterator类的模板定义中,有三个模板参数:T, Ref, 和 Ptr。这三个参数是为了提供更大的灵活性和控制。

//List的迭代器类
  template<class T, class Ref, class Ptr>
  class ListIterator
  {
    typedef ListNode<T>* PNode;
    typedef ListIterator<T, Ref, Ptr> Self;
  public:
    ListIterator(PNode pNode = nullptr):_pNode(pNode)
    {}
    ListIterator(const Self& l): _pNode(l._pNode)
    {}
    T& operator*()
    {
      return _pNode->_val;
    }
    T* operator->()
    {
      return &*this;
    }
    Self& operator++()
    {
      _pNode = _pNode->_pNext;
      return *this;
    }
    Self operator++(int)
    {
      Self temp(*this);
      _pNode = _pNode->_pNext;
      return temp;
    }
    Self& operator--()
    {
      _pNode = _pNode->_pPre;
      return *this;
    }
    Self& operator--(int)
    {
      Self temp(*this);
      _pNode = _pNode->_pPre;
      return temp;
    }
    bool operator!=(const Self& l)
    {
      return _pNode != l._pNode;
    }
    bool operator==(const Self& l)
    {
      return !(*this!=l);
    }
  private:
    PNode _pNode;
  };
相关文章
|
19天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
21 1
|
1月前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
49 7
|
1月前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
42 0
|
7月前
|
安全 Java
java线程之List集合并发安全问题及解决方案
java线程之List集合并发安全问题及解决方案
1097 1
|
6月前
|
Java API Apache
怎么在在 Java 中对List进行分区
本文介绍了如何将列表拆分为给定大小的子列表。尽管标准Java集合API未直接支持此功能,但Guava和Apache Commons Collections提供了相关API。
|
6月前
|
运维 关系型数据库 Java
PolarDB产品使用问题之使用List或Range分区表时,Java代码是否需要进行改动
PolarDB产品使用合集涵盖了从创建与管理、数据管理、性能优化与诊断、安全与合规到生态与集成、运维与支持等全方位的功能和服务,旨在帮助企业轻松构建高可用、高性能且易于管理的数据库环境,满足不同业务场景的需求。用户可以通过阿里云控制台、API、SDK等方式便捷地使用这些功能,实现数据库的高效运维与持续优化。
|
6月前
|
存储 安全 Java
详解Java中集合的List接口实现的ArrayList方法 | Set接口实现的HashSet方法
详解Java中集合的List接口实现的ArrayList方法 | Set接口实现的HashSet方法
|
7月前
|
Java API
使用 Java 来实现两个 List 的差集操作
使用 Java 来实现两个 List 的差集操作
255 3
|
6月前
|
存储 Java 索引
Java List接口实现原理与性能评估
Java List接口实现原理与性能评估
|
6月前
|
存储 缓存 安全
Java List操作详解及常用方法
Java List操作详解及常用方法
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等