【C++】STL——list模拟实现(2)

简介: 【C++】STL——list模拟实现(2)

五、list结构的完善

       上面我们对节点结构、正向与反向迭代器结构实现原理及注意点一一做了介绍,最后一步也是最重要的一步,那就是将list结构完善起来,实现list的功能。

1.构造函数

       list的成员变量是一个节点类,在构造头节点时,需要将这单个头节点构造成一个双向循环链表;

1ecd1b2606ed46e9956a89f231c9802c.png

//构造函数
list()
{
  _head = new Node;     //new一个节点出来
  _head->_prev = _head; 
  _head->_next = _head; //_prev 和 _next 同时指向了头结点,形成了双向循链表
}

2.拷贝构造

       拷贝构造是用一个已有对象去构造出另一个对象,首先将待构造对象进行初始化,然后利用迭代器区间去构造一个和lt1一样的临时的tmp对象,再进行数据的交换,达到深拷贝的目的。

//拷贝构造 --- 现代写法 lt2(lt1)
list(const list<T>& lt)
{
  _head = new Node;
  _head->_prev = _head;
  _head->_next = _head;
  list<T> tmp(lt.begin(), lt.end());
  std::swap(_head, tmp._head);
}

1ecd1b2606ed46e9956a89f231c9802c.png

3.迭代器区间构造

通过传过来的迭代器区间进行初始化

//迭代器区间构造
template<class IputIterator>
list(IputIterator first, IputIterator last)
{
  _head = new Node;
  _head->_prev = _head;
  _head->_next = _head;
  while (first != last)
  {
    push_back(*first);//尾插数据,会根据不同类型的迭代器进行调用
    ++first;
  }
}

4.用n个val构造

       通过用n个val来对对象进行初始化,需要注意这里的T( )是一个匿名对象,作为val的缺省参数,因为我们并不知道传给val的是一个对象还是一个整形(或其他),给缺省参数的好处在于,对于自定义类型编译器会去调用自定义类型的构造函数来对val进行初始化,如果是内置类型,它也是有自己的构造函数

//我们常常对内置类型的定义是这样的
int i = 10;
//但其实也可以这样定义
int i = int(10);
//用n个val个构造              
list(int n, const T& val = T())                
{                          
  _head = new Node;
  _head->_prev = _head;
  _head->_next = _head;
  for (int i = 0; i < n; i++)
  {
    push_back(val);
  }
}

对于用n个val进行初始化和迭代器区间初始化,起初我是这样写的(为了和库一致) ,测试时却出现问题:非法的间接寻址

//迭代器区间初始化
template<class IputIterator>
list(IputIterator first, IputIterator last){...}
//用n个val初始化
list(size_t n, const T& val = T()) {...}   
//测试日期类
struct Date
{
  int _year;
  int _month;
  int _day;
  Date(int year = 1, int month = 1, int day = 1)
    :_year(year)
    , _month(month)
    , _day(day)
  {}
};
void test_list4()
{
  list<Date> lt1(5, Date(2022, 6, 21));//1
  for (auto e : lt1)
  {
    cout << e._year << "/" << e._month << "/" << e._day << endl;
  }
  cout << endl;
  list<int> lt2(5, 1);//2
  for (auto e : lt2)
  {
    cout << e << " ";
  }
  cout << endl;
}

1ecd1b2606ed46e9956a89f231c9802c.png

5.赋值重载

//赋值重载 --- 现代写法 lt2 = lt1
list<T>& operator=(list<T> lt)
{
  std::swap(_head, lt._head);
  return *this;
}

6.析构函数

析构函数可以结合着erase函数看

~list()
{
  clear();
  delete _head;
  _head = nullptr;
}
void clear()
{
  iterator it = begin();
  while (it != end())
  {
    //写法1
    erase(it++); //it是一个正向迭代器,采用后置++传给erase
    //写法2
    iterator del = it++;
    delete del._node;
    //写法3
    iterator del = it++;
    erase(del);
  }
  _head->_prev = _head;
  _head->_next = _head;
}

1ecd1b2606ed46e9956a89f231c9802c.png

7.迭代器的实现

在list类中,我们typedef了正向与反向迭代器,解释一下其目的及作用

//正向迭代器    

typedef __list_iterator<T, T&, T*> iterator;
typedef __list_iterator<T, const T&, const T*> const_iterator;

//反向迭代器

typedef __list_reverse_iterator<iterator, T&, T*> reverse_iterator;
typedef __list_reverse_iterator<iterator, const T&, const T*>const_reverse_iterator;

  list是一个类模板,参数列表是T类型(未知类型)正向与反向迭代器分别都有三个参数,在list内重命名相当于内嵌定义,显示传参,我们对迭代器中需要返回值的地方不能进行类型的准确判断,通过这种显示传参,确定了迭代器三个参数的基本对于类型。


       对于正向迭代器,我们显示传参<T、T&、T*>如果是const迭代器,再重新typedef一个const迭代器;


       对于反向迭代器,我们是需要正向迭代器去构造,所有显示传参就可以给到<iterator, T&, T*>

namespace mlg
{
    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;
        iterator begin()
        {
          return iterator(_head->_next);//返回头节点的下一个结点
        }
        iterator end()
        {
          return iterator(_head);//返回头节点
        }
        const_iterator begin()const
        {
          return const_iterator(_head->_next);//返回头节点的下一个结点
        }
        const_iterator end()const
        {
          return const_iterator(_head);//返回头节点
        }
        //反向迭代器
        typedef __list_reverse_iterator<iterator, T&, T*> reverse_iterator;
        typedef __list_reverse_iterator<iterator, const T&, const T*>const_reverse_iterator;
        reverse_iterator rbegin()
        {
          return reverse_iterator(end());   //返回正向迭代器的结束位置
          //return reverse_iterator(_head); //等价与正向迭代器的头节点
        }
        reverse_iterator rend()
        {
          return reverse_iterator(begin());        //返回正向迭代器的起始位置
          //return reverse_iterator(_head->_next); //等价于正向迭代器头节点的下一个结点
        }
        const_reverse_iterator rbegin()const
        {
          return const_reverse_iterator(end());   //返回正向迭代器的结束位置
          //return const_reverse_iterator(_head); //等价与正向迭代器的头节点
        }
        const_reverse_iterator rend()const
        {
          return const_reverse_iterator(begin());        //返回正向迭代器的起始位置
          //return const_reverse_iterator(_head->_next); //等价于正向迭代器头节点的下一个结点
        }
    };
}

8.增删查改

//头插
void push_front(const T& x)
{
  insert(begin(), x);
}
//尾插
void push_back(const T& x)
{
  /*
  Node* tail = _head->_prev;
  Node* newnode = new Node(x);
  tail->_next = newnode;
  newnode->_prev = tail;
  newnode->_next = _head;
  _head->_prev = newnode;
  */
  insert(end(), x);
}
//头删
void pop_front()
{
  erase(begin());
}
//尾删
void pop_back()
{
  erase(--end());
}
//在pos位置插入元素x
iterator insert(iterator pos, const T& x)
{
  Node* cur = pos._node;    
  Node* prev = cur->_prev;     
  Node* newnode = new Node(x); 
  prev->_next = newnode;      
  newnode->_prev = prev;
  newnode->_next = cur;
  cur->_prev = newnode;
  return iterator(newnode);//在pos位置插入返回的是新插入的节点位置
}
//在pos位置删除元素
iterator erase(iterator pos)
{
  assert(pos != end());
  Node* prev = pos._node->_prev;
  Node* next = pos._node->_next;
  delete pos._node;
  prev->_next = next;
  next->_prev = prev;
  return iterator(next);//返回的是删除pos位置节点的下一个节点
}

六、完整代码

1.ListNode.h

#pragma once
namespace mlg
{
  template<class T>
  struct ListNode
  {
    ListNode<T>* _prev;
    ListNode<T>* _next;
    T _data;
    ListNode(const T& data= T())
      :_prev(nullptr)
      ,_next(nullptr)
      ,_data(data)
    {}
  };
}

2.iterator.h

#pragma once
namespace mlg
{
  template<class T,class Ref,class Ptr>
  struct __list_iterator
  {
    typedef ListNode<T> Node;
    typedef __list_iterator<T, Ref, Ptr> self;
    typedef Ref reference;
    typedef Ptr pointer;
    Node* _node;
    __list_iterator(Node* x)
      :_node(x)
    {}
    Ref operator*()
    {
      return _node->_data;
    }
    Ptr operator->()
    {
      return &_node->_data;
    }
    //++it
    self& operator++()
    {
      _node = _node->_next;
      return *this;
    }
    //it++
    self operator++(int)
    {
      self tmp(*this);
      _node = _node->_next;
      return tmp;
    }
    //--it
    self& operator--()
    {
      _node = _node->_prev;
      return *this;
    }
    //it--
    self operator--(int)
    {
      self tmp(*this);
      _node = _node->_prev;
      return tmp;
    }
    bool operator==(const self& it)const
    {
      return _node == it._node;
    }
    bool operator!=(const self& it)const
    {
      return _node != it._node;
    }
  };
}

3.reverse_iterator.h

#pragma once
namespace mlg
{
    template<class Iterator,class Ref,class Ptr>
  class __list_reverse_iterator
  {
    typedef __list_reverse_iterator<Iterator, Ref, Ptr> self;
  public:
    __list_reverse_iterator(Iterator it)
      :_it(it)
    {}
    Ref operator*()
    {
      Iterator prev = _it;
      return *--prev;
    }
    Ptr operator->() {return &operator*();}
    //++it
    self& operator++()
    {
      --_it;
      return *this;
    }
    //it++
    self operator++(int)
    {
      self tmp(*this);
      _it--;
      return tmp;
    }
    //--it
    self& operator--()
    {
      ++_it;
      return *this;
    }
    //it--
    self operator--(int)
    {
      self tmp(*this);
      _it++;
      return tmp;
    }
    bool operator!=(const self& rit)const {return _it != rit._it;}
    bool operator==(const self& rit)const {return _it == rit._it;}
  private:
    Iterator _it;
  };
}

4.list.h

#pragma once
#include "ListNode.h"
#include "iterator.h"
#include "reverse_iterator.h"
namespace mlg
{
  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;
    iterator begin() {return iterator(_head->_next);}
    iterator end() {return iterator(_head);}
    const_iterator begin()const {return const_iterator(_head->_next);}
    const_iterator end()const {return const_iterator(_head);}
    //反向迭代器
    typedef __list_reverse_iterator<iterator, T&, T*> reverse_iterator;
    typedef __list_reverse_iterator<iterator, const T&, const T*> const_reverse_iterator;
    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());}
    //构造函数
    list()
    {
      _head = new Node;     //new一个节点出来
      _head->_prev = _head; 
      _head->_next = _head; //_prev 和 _next 同时指向了头结点,形成了双向循链表
    }
    //用n个val个构造              
    list(int n, const T& val = T())                
    {                         
      _head = new Node;
      _head->_prev = _head;
      _head->_next = _head;
      for (int i = 0; i < n; i++)
      {
        push_back(val);
      }
    }
    //迭代器区间构造
    template<class IputIterator>
    list(IputIterator first, IputIterator last)
    {
      _head = new Node;
      _head->_prev = _head;
      _head->_next = _head;
      while (first != last)
      {
        push_back(*first);
        ++first;
      }
    }
    //拷贝构造 --- 现代写法 lt2(lt1)
    list(const list<T>& lt)
    {
      _head = new Node;
      _head->_prev = _head;
      _head->_next = _head;
      list<T> tmp(lt.begin(), lt.end());
      std::swap(_head, tmp._head);
    }
    //赋值重载 --- 现代写法 lt2 = lt1
    list<T>& operator=(list<T> lt)
    {
      std::swap(_head, lt._head);
      return *this;
    }
    ~list()
    {
      clear();
      delete _head;
      _head = nullptr;
    }
    void clear()
    {
      iterator it = begin();
      while (it != end())
      {
        erase(it++);
      }
      _head->_prev = _head;
      _head->_next = _head;
    }
    void push_front(const T& x){ insert(begin(), x); }
    void push_back(const T& x) { insert(end(), x);}
    void pop_front() {erase(begin());}
    void pop_back(){erase(--end());}
    //在pos位置插入元素x
    iterator insert(iterator pos, const T& x)
    {
      Node* cur = pos._node;
      Node* prev = cur->_prev;
      Node* newnode = new Node(x);
      prev->_next = newnode;
      newnode->_prev = prev;
      newnode->_next = cur;
      cur->_prev = newnode;
      return iterator(newnode);
    }
    //在pos位置删除元素
    iterator erase(iterator pos)
    {
      assert(pos != end());
      Node* prev = pos._node->_prev;
      Node* next = pos._node->_next;
      delete pos._node;
      prev->_next = next;
      next->_prev = prev;
      return iterator(next);
    }
  private:
    Node* _head;
  };
  void print_list(const list<int>& lt)
  {
    list<int>::const_iterator it = lt.begin();
    while (it != lt.end())
    {
      cout << *it << " ";
      ++it;
    }
    cout << endl;
  }
  void test_list5()
  {
    list<int> lt;
    lt.push_back(1);
    lt.push_back(2);
    lt.push_back(3);
    lt.push_back(4);
    list<int>::reverse_iterator rit = lt.rbegin();
    while (rit != lt.rend())
    {
      cout << *rit << " ";
      ++rit;
    }
    cout << endl;
  }
}

        以上是对list模拟实现的全部结束,大家可以在.c文件下进行测试。如果存在任何问题,或者有不同的见解,大家可以直接在评论区提出来


目录
相关文章
|
22天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
26 1
|
1月前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
51 7
|
1月前
|
存储 编译器 C++
C++ initializer_list&&类型推导
在 C++ 中,`initializer_list` 提供了一种方便的方式来初始化容器和传递参数,而右值引用则是实现高效资源管理和移动语义的关键特性。尽管在实际应用中 `initializer_list&&` 并不常见,但理解其类型推导和使用方式有助于深入掌握现代 C++ 的高级特性。
23 4
|
2月前
|
存储 编译器 C语言
【c++丨STL】vector的使用
本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
96 4
|
2月前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
101 5
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
78 2
|
2月前
|
存储 算法 Linux
【c++】STL简介
本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
82 0
|
1月前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
46 0
|
11天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
51 18
|
11天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
37 13