【c++】list模拟实现(1)

简介: 【c++】list模拟实现(1)

list的接口

#pragma once
#include<iostream>
#include<assert.h>
using namespace std;
namespace zjw
{
  template<class T>
  struct listnode    
  {
    listnode* <T>_next;
    listnode* <T>_prev;
    T _data;
    listnode(const T& x = T())
      :_prev(nulllptr)
      ,_next(nullptr)
      ,_data(x)
    {
    
    
    
    
    }
  };
  template<class T>
  struct _list_iterator
  {
    typedef listnode <T>node;
    
    typedef _list_iterator<T>self;
    node* _node;
    _list_iterator(node* n)
      :_node(n)
    {
    }
    self& operator++()
    {
    }
    self& operator--()
    {
    }
    self operator++(int)
    {
    }
    self operator--(int)
    {
    }
    bool operator!=(const self& s)
    {
    
    }
    bool operator==(const self& s)
    {
    }
    T& operator()
    {
    
    }
  };
     template <class T>
   class list
   {    typedef listnode <T>node;
    public:
     typedef _list_iterator<T>  iterator;
     list()
     {
       empty_init();
     
     }
     void empty_init()
     {
     
     
     }
     iterator begin()
     {
     
     }
     iterator end()
     {
     
     
     
     }
     void clear()
     {
     
     
     }
     ~list
     {
     
     
     }
     list(const list<T>& it)
     {
     
     
     }
     void push_back(const T&x)
     {
     
     
     
     }
     void push_front(const T& x)
     {
     }
     void pop_back()
     {
     
     
     
     }
     void pop_front()
     {
     
     
     
     
     }
     void swap(list<T>& tmp)
     {
       
     }
     list<T>& operator=(list<T>it)
     {
     
     
     
     }
     iterator insert(iterator pos ,const T&x)
     {
     
     
     
     
     
     }
     iterator erase(iterator pos)
     {
     
     
     
     
     }
   private :
     node* _head;
   };
}

迭代器前置++,前置–,后置++,后置–

self& operator++()
    { 
      _node= _node->next;
      return * this;
    }
    self& operator--()
    {
      _node = _node->_prev;
      return * this;
    }
    self operator++(int)
    {
      self tmp(*this);
      _node = _node->_next;
      return tmp;
    }
    self operator--(int)
    {
      self tmp(*this);
      _node = _node->_prev;
      return tmp;
      
    }

迭代器重载等于,以及不等于,以及解引用访问数据

bool operator!=(const self& s)
    {
      return s._node != _node;
    
    }
    bool operator==(const self& s)
    {
      return s._node == _node;
    }
    T& operator*()
    {
      return _node->_data;
    
    }
    

链表的默认构造,以及尾插,获取链表的头和尾巴,以及测试尾插函数

list()
     {
       empty_init();
     
     }
     void empty_init()
     {
       _head = new node();
       _head->_prev = _head;
       _head->_next = _head;
     
     
     }
     iterator begin()
     {
       return _head->_next;
     
     }
     iterator end()
     {
       return _head;
     
     
     }
       void push_back(const T&x)
     {
       node* newnode = new node(x);
       node* tail = _head->_prev;
       newnode->_next = _head;
       newnode->_prev = tail;
       tail->_next = newnode;
       _head->_prev = newnode;
       
     
     
     
     }

尾插测试,以及迭代器测试

void test1()
   {
     list<int>res;
     res.push_back(1);
     res.push_back(2);
     res.push_back(3);
     res.push_back(4);
     list<int> ::iterator it = res.begin();
     while (it != res.end())
     {
       cout <<*it << " ";
       it++;
      }
 
   }


const迭代器的实现

我们就可以新增两个带const的begin(),end(),权限平移就可以调动。

iterator begin() const
     {
       return _head->_next;
     }
     iterator end() const
     {
       return _head;
     }

如果我们要修改迭代器所指向位置的值呢??

我们应该如何模拟实现一个常量迭代器呢??

如果按照以下的方式,可以实现吗??

所以我们应该怎么实现呢??

迭代器中的解引用函数,我们需要让他的返回值不被修改,所以 这个函数的返回值加const 就好了,所以我们在实现一个类,这个类基本和普通迭代器的类一样,只是在解引用函数上有区分。

template<class T>
  struct const_list_iterator
  {
    typedef    listnode <T>  node;
    typedef const_list_iterator<T> self;
    node* _node;
    const_list_iterator(node* n)
      :_node(n)
    {
    }
    self& operator++()
    {
      _node = _node->_next;
      return *this;
    }
    self& operator--()
    {
      _node = _node->_prev;
      return *this;
    }
    self operator++(int)
    {
      self tmp(*this);
      _node = _node->_next;
      return tmp;
    }
    self operator--(int)
    {
      self tmp(*this);
      _node = _node->_prev;
      return tmp;
    }
    bool operator !=(const self& s)
    {
      return s._node != _node;
    }
    bool operator==(const self& s)
    {
      return s._node == _node;
    }
    const T operator*()
    {
      return _node->_data;
    }
  };

这样子,就实现了const 的迭代器,但是重新搞一个类好麻烦呀,于是有大佬就想出了在提供一个模板参数来解决这个问题


->运算符重载函数

我们可以想一下什么时候会用到->,当指针it指向的是结构体的话,我们可以访问使用

it->data,来访问数据,也可先对指针解引用*it,*it表示拿到这个结构体,然后使用.来访问(*it).data;

T*  operator->()
    {
      return &_node->_data;
    }
struct AA
   {
     int a1;
     int a2;
     AA(int _a1=0,int _a2=0)
       :a1(_a1)
       ,a2(_a2)
     
     {
     
     }
   };

测试->

void test3()
   {
     list<AA>res; 
     res.push_back(AA(1,1));
     res.push_back(AA(2, 2));
     res.push_back(AA(3, 3));
     res.push_back(AA(4, 4));
     list<AA> ::iterator it = res.begin();
     while (it != res.end())
     {
       cout << it.operator->()->a1<<":" << it.operator->()->a2 << endl;
       it++;
   }
 
   
   }


这里有一个问题就是->访问也是取数据,但取到的数据能修改吗?这就面临和解引用取数据同样的问题。

我们现在需要做的是const迭代器的不能被将修改,普通的可以修改值

我发现这里和解引用那里不一样的是,解引用那里是值不可被修改,这里是地址不可被修改


=运算符重载赋值

void swap(list<T>& tmp)
     {
       std::swap(_head, tmp._head);
     }
     list<T>& operator=(list<T>it)
     {
       swap(it);
       return *this;
     
     
     
     }

赋值函数测试

void test4()
   {
    
     list<int>res;
     res.push_back(1);
     res.push_back(2);
     res.push_back(3);
     res.push_back(4);
     list<int> ret = res;
     list<int> ::iterator it = ret.begin();
     while (it != ret.end())
     {
       cout << *it << " ";
       it++;
     }
    
   
   
   
   }



目录
相关文章
|
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++ initializer_list&&类型推导
在 C++ 中,`initializer_list` 提供了一种方便的方式来初始化容器和传递参数,而右值引用则是实现高效资源管理和移动语义的关键特性。尽管在实际应用中 `initializer_list&&` 并不常见,但理解其类型推导和使用方式有助于深入掌握现代 C++ 的高级特性。
22 4
|
3月前
|
存储 搜索推荐 C++
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器2
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器
75 2
|
3月前
|
存储 算法 C++
【C++打怪之路Lv10】-- list
【C++打怪之路Lv10】-- list
28 1
|
3月前
|
存储 C++ 容器
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器1
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器
78 5
|
3月前
|
存储 编译器 C++
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
94 2
|
3月前
|
C++
【C++】C++ STL 探索:List使用与背后底层逻辑(三)
【C++】C++ STL 探索:List使用与背后底层逻辑
|
3月前
|
C++
【C++】C++ STL 探索:List使用与背后底层逻辑(二)
【C++】C++ STL 探索:List使用与背后底层逻辑
|
3月前
|
存储 编译器 C++
【C++】C++ STL 探索:List使用与背后底层逻辑(一)
【C++】C++ STL 探索:List使用与背后底层逻辑