欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 汽车 > 时评 > list底层详解

list底层详解

2025/7/4 4:55:47 来源:https://blog.csdn.net/2301_80418172/article/details/141576021  浏览:    关键词:list底层详解

目录

介绍

list的实现 

1.自定义节点

2.迭代器封装

构造函数

前置++和后置++

前置--和后置--

*操作符和->操作符

==和!=操作符

iterator和const_iterator

3. list类

构造函数和析构函数

=赋值操作

头尾迭代器

插入和删除

头插头删尾插尾删

list接口函数总代码


介绍

1. list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。
2. list的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。
3. list与forward_list非常相似:最主要的不同在于forward_list是单链表,只能朝前迭代,已让其更简单高效。
4. 与其他的序列式容器相比(array,vector,deque),list通常在任意位置进行插入、移除元素的执行效率更好。
5. 与其他序列式容器相比,list和forward_list最大的缺陷是不支持任意位置的随机访问,比如:要访问list的第6个元素,必须从已知的位置(比如头部或者尾部)迭代到该位置,在这段位置上迭代需要线性的时间开销;list还需要一些额外的空间,以保存每个节点的相关联信息(对于存储类型较小元素的大list来说这可能是一个重要的因素)

List原文档,我们可以去cplusplus网站去查看详情

https://cplusplus.com/reference/list/list/?kw=list

list的实现 

1.自定义节点

因为是双向链表,所以我们只需要定义前指针和后指针,以及该节点的值data,因为我们以后要经常访问里面的内容所以使用结构体更合适.另外需要一个默认构造函数来方便以后的拷贝和初始化;

template<class T>
struct Node
{struct Node* next;struct Node* prev;  T data;Node(const T& val=T()):prev(nullptr), next(nullptr), data(val)//初始化{}
};

2.迭代器封装

迭代器作为STL中一个万能的访问方式,在任何一个容器中都是可以使用++,*像这样的操作的,我们之前在vector中iterator是对T*进行别名的,因为vector是连续性的容器,所以T*是可以++来进行遍历操作的,但是list是链表,其地址是不连续的,所以无法直接对迭代器进行++等操作,因此我们需要对Node* 和多个重载运算符来封装成iterator 和const iterator;

迭代器包装的还是T* ,在这里是Node*,后面++操作需要对迭代器本身处理,所以我们也把listiterator<T>简化成self; 

构造函数

	//封装迭代器  template<class T>class listiterator{typedef Node<T> node;typedef listiterator <T>   self;public:listiterator(node* node)   //拷贝节点  迭代器还是node* 只是多添加了些运算符操作:_node(node){}node* _node;};

iterator只需要一个拷贝构造函数即可;直接将_node初始化为node;

前置++和后置++

这个迭代器本身就是一个类,所以自增后返回的是一个类,这就用到了self,前置++:直接将_node指针后移指向next即可;然后返回*this;后置的就需要一个中间变量来储存操作前的迭代器,然后返回临时变量,需要注意的是后置++返回的是临时变量,因此返回的类型不能使用引用.

	//前置++self operator ++(){_node = _node->next;return *this;}//后置++self  operator++(int){self  tmp(*this);_node = _node->next;return tmp;}

前置--和后置--

与上面的操作模版相同,返回的是上一个节点;

	//前置--self& operator --(){_node = _node->prev;return *this;}//后置--self  operator --(int){self tmp(*this);_node = _node->prev;return tmp;}

*操作符和->操作符

解引用操作直接返回节点的值即可,这里比较难以理解的就是->返回的是值data的地址,这里调用时省略了一个operator->(),因为迭代器本身就是一个类重载的操作符->指向的就是_node,然后_node才能访问data,这里这样写可以直接访问到data;

		//*迭代器T& operator*(){return _node->data;}//->迭代器T* operator->(){return &_node->data;}

==和!=操作符

这里直接比较就行了;

bool operator !=(const self& it)const
{return it._node != _node;
}
//迭代器相等
bool operator ==(const self& it)const
{return it._node == _node;
}

iterator和const_iterator

我们知道const_iterator是不能够改变内部的值的,所以我们为了写出一个const_iterator直接通过模版来让编译器来写;其中与能够改变内部值的只有*和->操作,我们只需要把返回值改成const T&和const T*就能实现常迭代器,为此,我们直接将这两个函数的返回值定义为模版ref和ptr;这样就能轻松完成两种迭代器了;

template<class T,class ref,class ptr>
class listiterator
{typedef Node<T> node;  typedef listiterator <T ,ref,ptr>   self;            
public:  //*迭代器ref operator*(){return _node->data;                    } //->迭代器ptr  operator->() {return &_node->data;            }
}

3. list类

list的源码中私有成员是只有一个节点哨兵位_head的;我们先来把封装好的节点和迭代器造个别名再实现下构造函数;

构造函数和析构函数

因为构造函数和拷贝构造函数都需要先创造一个哨兵位,所以我们把哨兵位的创建写成函数.

拷贝构造函数直接遍历形参调用尾插push_back即可;

析构函数需要挨个的把所有的节点释放掉,最后释放掉哨兵位;这里消除节点我们可以直接用后面实现的erase;

template<class T>  
class list
{typedef Node<T>  node;//节点
public:typedef listiterator<T,T&,T*>  iterator;  //迭代器typedef listiterator  <T, const T&, const T*>const_iterator;   //常量迭代器//创建哨兵位void init(){_head = new node();                  _head->next = _head->prev =_head ;                                                                          }//构造函数list()   {init();   }//拷贝构造函数list(const list<T>&it){init();       for (const auto& x : it){push_back(x);     }}
//析构函数
~list()
{clear(); delete _head;   _head = nullptr;   
}
//销毁函数
void clear()
{auto it = begin();      while (it != end()){it = erase(it);    }
}

=赋值操作

将一个类赋值给另一个类,我们现代的写法就是,形参不引用,使用swap与形参交换数据.

//list赋值操作
list<T>& operator=(list<T>  x)     
{swap(_head, x._head);  //直接交换哨兵位return *this;   
}

头尾迭代器

//头迭代器
iterator  begin()              //it是局部变量,返回值不能引用   
{iterator it(_head->next);    return it;   
}
//常量迭代器
const_iterator  begin()const
{const_iterator it(_head->next);return it;
}
//尾迭代器
iterator  end()
{iterator it(_head);return it;  
}
//常量尾迭代器
const_iterator  end()const
{const_iterator it(_head);   return it;  
}

插入和删除

插入就比较简单了并且没有迭代器失效的风险,改变对应两个前后两个节点的指向就行了.虽然不需要返回迭代器,但是最好按照文档来;
删除需要注意的是需要将删掉的节点释放掉,返回pos原来节点的下一个位置的迭代器;

//插入
iterator     insert(iterator pos, T x)
{node* cur = pos._node;       node* net = cur->next;    node* newnode = new node(x);    cur->next = newnode;newnode->prev = cur;newnode->next = net;net->prev = newnode;   return iterator(newnode);      //返回迭代器     
}
//删除
iterator  erase(iterator  pos)
{assert(pos  !=  end());                                                   node* cur = pos._node;   node* pre = cur->prev;   node* net = cur->next;    pre->next = net;     net->prev = pre;delete cur;  return iterator(net);
}

头插头删尾插尾删

直接调用对应的insert和erase,begin()就是_head后面的第一个有效数据,end()就是哨兵位,所以我们需要--一下,回到尾节点;

		//头插void push_front(T x){insert(begin(), x); }//头删void pop_front()  {erase(begin());}//尾插void push_back(T x)  {insert(--end(), x);  }//尾删void pop_back(){erase(--end());    }

list接口函数总代码

#pragma once 
#include<iostream>
#include<assert.h>
#include<algorithm>
using namespace std;
namespace bit
{template<class T>struct Node{struct Node* next;struct Node* prev;  T data;Node(const T& val=T()):prev(nullptr), next(nullptr), data(val)//初始化{}};
//封装迭代器  
template<class T,class ref,class ptr>
class listiterator
{typedef Node<T> node;  typedef listiterator <T ,ref,ptr>   self;            
public:  listiterator(node* node)   //拷贝节点  迭代器还是node* 只是多添加了些运算符操作:_node(node)               {}//前置++self operator ++(){_node = _node->next;        return *this;   }//后置++self  operator++(int){self  tmp(*this);_node = _node->next;return tmp;         }//前置--self& operator --()  {_node = _node->prev;    return *this;     }    //后置--self  operator --(int){self tmp(*this);_node = _node->prev;      return tmp;   }//*迭代器ref operator*(){return _node->data;                    } //->迭代器ptr  operator->() {return &_node->data;            }//bool operaotr != ( const self& it)              //{//	return _node != iit.    //}//迭代器不相等bool operator !=(const self& it)const   { return it._node != _node;                  }//迭代器相等bool operator ==(const self& it)const {return it._node == _node;    }node* _node;    
};
//list实现template<class T>  class list{typedef Node<T>  node;//节点public:typedef listiterator<T,T&,T*>  iterator;  //迭代器typedef listiterator  <T, const T&, const T*>const_iterator;   //常量迭代器//创建哨兵位void init(){_head = new node();                  _head->next = _head->prev =_head ;                                                                          }//构造函数list()   {init();   }//拷贝构造函数list(const list<T>&it){init();       for (const auto& x : it){push_back(x);     }}//析构函数~list(){clear(); delete _head;   _head = nullptr;   }//销毁函数void clear(){auto it = begin();      while (it != end()){it = erase(it);    }}//list赋值操作list<T>& operator=(list<T>  x)     {swap(_head, x._head);          return *this;   }//头迭代器iterator  begin()              //加不加引用都可以,加引用就是引用的常量;不加引用就是拷贝的常量    {iterator it(_head->next);    return it;   }//常量迭代器const_iterator  begin()const{const_iterator it(_head->next);return it;}//尾迭代器iterator  end(){iterator it(_head);return it;  }//常量尾迭代器const_iterator  end()const{const_iterator it(_head);   return it;  }void print(){node* cur = _head->next;   while (cur != _head){cout << cur->data<<" ";cur = cur->next;       }cout << endl; }//插入iterator     insert(iterator pos, T x){node* cur = pos._node;       node* net = cur->next;    node* newnode = new node(x);    cur->next = newnode;newnode->prev = cur;newnode->next = net;net->prev = newnode;   return iterator(newnode);      //返回迭代器     }//删除iterator  erase(iterator  pos){assert(pos  !=  end());                                                   node* cur = pos._node;   node* pre = cur->prev;   node* net = cur->next;    pre->next = net;     net->prev = pre;delete cur;  return iterator(net);}//头插void push_front(T x){insert(begin(), x); }//头删void pop_front()  {erase(begin());}//尾插void push_back(T x)  {insert(--end(), x);  }//尾删void pop_back(){erase(--end());    }private:node* _head;   };void test(){bit::list<int>t;t.push_back(1);t.push_back(2);t.push_back(3);t.push_back(4);   t.push_back(5);t.print();bit::list<int>t2;t2 = t;auto it = t2.begin();   //验证赋值操作t2.print();bit::list<int>t3(t2);auto it3 = t3.begin();while (it3 != t3.end()){cout << *it3 << " ";it3 = t3.erase(it3);}/*it++;cout << typeid(it).name()<<*it<<endl;auto it2 = t.end();cout << typeid(it2).name()<<*it << endl;bit::list<int>::iterator ita = t.begin();cout << *(it++) << endl; */}
}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词