news 2026/4/16 15:24:45

list的实现和使用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
list的实现和使用

list深入讲解

1. 简述与适用场景

list是双向链表的标准实现,适用于:

  • 频繁在容器中间进行插入/删除的场景(已知位置的情况下这些操作为 O(1))。
  • 需要稳定的指针/迭代器(对于不被删除的元素,list的迭代器在大多数操作后仍然有效)。

不适合场景:

  • 需要频繁随机访问(operator[]不可用,跳转到第 n 个元素需 O(n))。
  • 大量排序且希望用sortsort需要随机访问迭代器)。
cpp list<int>lt;lt.push_back(1);lt.push_back(3);lt.push_back(2);lt.push_back(6);for(autoe:lt)cout<<e<<" ";

2. 迭代器分类(复盘)

按功能:iteratorconst_iteratorreverse_iterator等。
按性质(底层决定):

  • 单向:forward_list(仅支持++
  • 双向:list(支持++--
  • 随机访问:vector/deque(支持+/-

关键点:算法要求特定迭代器类别(例如sort要求随机访问迭代器),因此不能直接对list使用sort

3. 增删改查基础

  • push_back/push_front:在尾/头插入元素。
  • emplace_back/emplace_front:直接在容器内部构造对象,避免额外拷贝/移动。

emplace_back可以代替push_back使用且在默写场景下效率更高。

cppstructA{A(inta1=1,inta2=2):_a1(a1),_a2(a2){}int_a1;int_a2;};list<A>lt1;Aaa1(1,1);lt1.push_back(aa1);lt1.emplace_back(3,3);// 直接构造
  • insert(it, value):在迭代器it之前插入(O(1))。
  • erase(it):删除it指向的节点(返回下一个迭代器)。
  • remove(value):查找并删除所有等于value的元素(按值删除,不需要迭代器)。
  • splice(...):在常数时间内把一个链表或一段节点从一个list移动到另一个list(不拷贝节点,仅变指针)。
    • lt1.splice(it, lt2):把lt2全部插入到lt1it前,lt2变空。
    • lt.splice(lt.begin(), lt, it):把单个节点移动到头部。
    • lt.splice(lt.begin(), lt, it, lt.end()):把[it, end)段移动到头部。
  • reverse():链表自身提供反转(lt.reverse())。
    • 注意:std::reverse(lt.begin(), lt.end())对双向迭代器同样可用,但list有专门的reverse()

4. 排序与合并

  • std::sort不能直接用于list(需随机访问迭代器)。
  • std::list提供list::sort()(内部是归并排序,稳定,适合链表)。
    • 可以传入仿函数或函数对象,如lt.sort(greater<int>())进行降序。
  • 如果希望利用std::sort的速度(随机访问算法常常更快),可以先把list内容拷贝到vector,排序后再assignlist
cpp vector<int>v(lt.begin(),lt.end());sort(v.begin(),v.end());lt.assign(v.begin(),v.end());
  • merge:合并两个已排序的list(将元素从参数表list中移动到目标list),操作后被合并的list变空。
cpp x.sort();y.sort();x.merge(y);// y 变空,x 成为合并后的有序列表

5. 性能与复杂度总结(常见操作)

  • push_back,push_front: O(1)
  • insert(在已知位置): O(1)
  • erase(给定迭代器): O(1)
  • find(基于值): O(n)
  • splice:O(1)(只是改变指针,不拷贝)
  • sort()list::sort): O(n log n)(归并排序)
  • remove(value): O(n)

注意:虽然对单个已知位置的插入/删除为 O(1),但查找位置仍然可能需要 O(n)。

6. 常见“坑”与建议

  • 不能对list使用std::sort(会编译错误);应使用list::sort或先复制到vector
  • splice非常高效,但要注意源list中被移动节点的迭代器/引用在语义上仍然有效并且指向原节点(但被移走后会在新容器中可用)。
  • list执行大量随机访问(步进迭代器若干次以到达位置)会很慢,应考虑vectordeque
  • emplace_back在构造复杂对象时通常比push_back更高效,因为能避免不必要的拷贝/移动。
  • 在并发环境下list不提供线程安全,操作前需加锁或采用其他并发容器。

7. 小结

  • std::list在需要频繁、局部化的插入和删除时是优秀的选择,并通过splicemerge等操作能以常数时间移动节点。
  • 若需随机访问或使用需要随机访问迭代器的算法(如std::sort),应使用vector/deque或将list的数据转换到vector再处理。
  • 选择容器时优先考虑操作模式:插入/删除 vs 随机访问 vs 内存紧凑性与缓存友好性(vector通常更缓存友好)。

list模拟实现

namespacelist{template<classT>classlist_node{public:T _data;list_node<T>*_next;list_node<T>*_prev;list_node(constT&data=T()):_data(data),_next(nullptr),_prev(nullptr){}};//重点:迭代器的实现template<classT>structlist_iterator//默认{typedeflist_node<T>Node;typedeflist_iterator<T>Self;//const iterator -> 迭代器本身不能修改//const_iterator -> 指向内容不能修改Node*_node;T&operator*(){return_node->_data;}Self&operator++()//前置{_node=_node->next;return*this;}Self&operator++(int)//后置{Selftmp(*this);_node=_node->next;returntmp;}Self&operator--(){_node=_node->_prev;return*this;}Self&operator--(int){Selftmp(*this);_node=_node->prev;returntmp;}T&operator->(){return_node->_data;}booloperator!=(constSelf&s)const{return_node!=s._node;}list_iterator(Node*node):_node(node){}};//const迭代器的实现template<classT>structlist_const_iterator//默认{typedeflist_node<T>Node;typedeflist_const_iterator<T>Self;Node*_node;list_const_iterator(Node*node):_node(node){}constT&operator*(){return_node->_data;}Self&operator++()//前置{_node=_node->next;return*this;}Self&operator++(int)//后置{Selftmp(*this);_node=_node->next;returntmp;}Self&operator--(){_node=_node->_prev;return*this;}Self&operator--(int){Selftmp(*this);_node=_node->prev;returntmp;}constT*operator->(){return_node->_data;}booloperator!=(constSelf&s)const{return_node!=s._node;}};template<classT>classlist{typedeflist_node<T>Node;public:typedeflist_iterator<T,T&,T*>iterator;typedeflist_iterator<T,constT&,constT*>const_iterator;list():_head(newNode()),_size(0){_head->_next=_head;_head->_prev=_head;}~list(){clear();delete_head;}list(constlist<T>&lt):list(){for(autoconst&e:lt)push_back(e);}list<T>&operator=(list<T>lt){swap(lt);return*this;}voidswap(list<T>&lt){std::swap(_head,lt._head);std::swap(_size,lt._size);}iteratorbegin(){returniterator(_head->_next);}iteratorend(){returniterator(_head);}const_iteratorbegin()const{returnconst_iterator(_head->_next);}const_iteratorend()const{returnconst_iterator(_head);}voidpush_back(constT&x){Node*newnode=newNode(x);Node*tail=_head->_prev;tail->_next=newnode;newnode->_prev=tail;newnode->_next=_head;_head->_prev=newnode;++_size;}iteratorinsert(iterator pos,constT&x){Node*cur=pos._node;Node*prev=cur->_prev;Node*newnode=newNode(x);newnode->_next=cur;cur->_prev=newnode;newnode->_prev=prev;prev->_next=newnode;++_size;returniterator(newnode);}iteratorerase(iterator pos){assert(pos!=end());Node*cur=pos._node;Node*prev=cur->_prev;Node*next=cur->_next;prev->_next=next;next->_prev=prev;deletecur;--_size;returniterator(next);}voidpop_back(){assert(!empty());erase(iterator(_head->_prev));}voidpop_front(){assert(!empty());erase(begin());}voidclear(){Node*cur=_head->_next;while(cur!=_head){Node*next=cur->_next;deletecur;cur=next;}_head->_next=_head;_head->_prev=_head;_size=0;}size_tsize()const{return_size;}boolempty()const{return_size==0;}private:Node*_head;size_t _size;};}

其中,迭代器的实现部分我们可以使用高度相似的模板复用实例化来优化代码

template<classT,classRef,classPtr>structlist_iterator//默认{typedeflist_node<T>Node;typedeflist_iterator<T,Ref,Ptr>Self;Node*_node;list_const_iterator(Node*node):_node(node){}Refoperator*(){return_node->_data;}Self&operator++()//前置{_node=_node->next;return*this;}Self&operator++(int)//后置{Selftmp(*this);_node=_node->next;returntmp;}Self&operator--(){_node=_node->_prev;return*this;}Self&operator--(int){Selftmp(*this);_node=_node->prev;returntmp;}Ptroperator->(){return_node->_data;}booloperator!=(constSelf&s)const{return_node!=s._node;}};
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 10:56:53

高配不高价!傲风G5凭实力入选入门级电竞椅推荐榜单

在办公与电竞场景日益融合的当下&#xff0c;一把能够兼顾人体工学支撑与多场景适配的座椅&#xff0c;已成为职场人士与电竞玩家共同追求的理想装备。傲风作为深耕电竞外设领域的专业品牌&#xff0c;连续六年稳居中国电竞椅销量榜首&#xff0c;不仅长期合作LPL、VCT等顶级赛…

作者头像 李华
网站建设 2026/4/16 10:57:04

网络知识要点:从入门到精通的基石指南

无论是软件开发、系统运维还是日常技术应用&#xff0c;网络知识都是不可或缺的底层支柱。理解数据如何在网络中穿梭&#xff0c;是解决复杂问题、设计高效系统的基础。本文将从底层到上层&#xff0c;梳理关键的网络知识要点。一、网络基石&#xff1a;核心概念与模型1. 核心目…

作者头像 李华
网站建设 2026/4/16 10:44:56

HarmonyOS应用代码混淆技术方案,为你的应用安全保驾护航

概述代码混淆技术可以增加代码的复杂性和模糊性&#xff0c;从而提高攻击者分析代码的难度。代码混淆有以下几个方面的作用&#xff1a;1. 保护知识产权&#xff1a;代码混淆防止他人轻易复制和窃取软件代码&#xff0c;增加逆向工程难度。2. 防止逆向工程&#xff1a;逆向工…

作者头像 李华
网站建设 2026/4/16 14:27:46

3分钟完成OpenSSL安装:极速方案对比

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个OpenSSL安装效率对比工具&#xff0c;功能包括&#xff1a;1. 传统源码编译方式 2. 包管理器安装&#xff08;apt/yum/brew&#xff09;3. Docker容器方案 4. 二进制包直接…

作者头像 李华
网站建设 2026/4/16 11:11:34

以龙企招为例,浅谈鸿蒙应用开发者激励计划 2025 参与心得

2025 年&#xff0c;我们带着 “龙企招” 鸿蒙应用&#xff0c;报名参与了鸿蒙应用开发者激励计划。原本满怀期待地提交上架申请&#xff0c;却收到了审核未通过的通知。这次经历虽有遗憾&#xff0c;却让我们深刻体会到鸿蒙生态对应用质量的严格要求&#xff0c;也为我们的应用…

作者头像 李华
网站建设 2026/4/16 12:21:07

Java安全机制入门:SecurityManager详解

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个面向Java初学者的SecurityManager教学项目&#xff0c;包含&#xff1a;1. 基础概念图解&#xff1b;2. 5个渐进式代码示例&#xff1b;3. 交互式练习&#xff08;修复预设…

作者头像 李华