C++中的List容器用法详解

文章目录

  • C++中的List容器用法详解
  • List 的特点
  • List 的重要接口用法介绍
    • 1.创建和初始化List
      • list
    • 2.插入元素
      • push_back
      • push_fornt
      • insert
    • 删除元素
      • pop_back
      • pop_font
      • clear
      • erase
    • 遍历List
      • 迭代器遍历
      • 范围for遍历
    • 排序List
      • sort
    • 反转List
      • reverse
    • 转移List
      • splice
    • 去重
      • unique
    • 合并
      • merge
    • 总结

C++中的List容器用法详解

在C++标准模板库(STL)中,list 是一种高效的、双向的、动态的容器,它允许在任何位置快速插入和删除元素。list 容器在内部通过节点来存储数据,每个节点都包含数据部分和指向前后节点的指针。这种设计使得 list 在进行插入和删除操作时具有较高的效率,因为只需要修改指针的指向,而不需要移动元素。

List 的特点

std::list 是 C++ 标准模板库(STL)中的一个容器,它具有以下特点:

  1. 双向链表list 是一个双向链表,每个元素都包含一个数据部分和两个指针,分别指向下一个元素和上一个元素。这种结构使得在链表的任何位置插入或删除元素都非常高效。

  2. 动态大小list 容器的大小是动态的,可以随着元素的添加和删除而自动调整。它不需要预先分配固定大小的内存。

  3. 不连续内存分配:与数组或 std::vector 不同,list 的元素不存储在连续的内存块中。每个元素都是独立分配的,这使得 list 在处理大量数据时能够提供更好的内存利用率和性能。

  4. 高效插入和删除:由于其链表结构,list 在任何位置插入或删除元素的时间复杂度都是常数时间(O(1)),这比 std::vector 在中间位置插入或删除元素的线性时间(O(n))要快得多。

  5. 不支持随机访问:由于 list 是一个链表,它不支持像数组或 std::vector 那样的随机访问。访问 list 中的元素需要从头开始遍历,因此其时间复杂度为线性时间(O(n))。

  6. 迭代器支持list 提供了双向迭代器,这意味着可以从两个方向遍历 list,但不支持随机访问迭代器。

  7. 内存管理list 的内存管理是自动的,当元素被插入时,list 会自动分配内存;当元素被删除时,list 会自动释放内存。

  8. 不支持直接访问:不能通过索引直接访问 list 中的元素,必须使用迭代器。

  9. 不支持 at() 方法list 不提供 at() 方法来访问元素,因为这需要随机访问能力,而 list 不支持随机访问。

  10. 不支持 [] 运算符:与 at() 方法类似,list 也不支持 [] 运算符来访问元素,原因同上。

  11. 不支持 resize() 方法list 不提供 resize() 方法来改变容器的大小,因为 list 的大小是动态的,且不连续分配内存。

  12. 不支持 reserve() 方法list 不提供 reserve() 方法来预分配内存,因为 list 的内存分配策略是自动的。

list 的这些特点使得它在需要频繁插入和删除操作的场景中非常有用,但在需要快速随机访问的场景中则不如 std::vector 或数组高效。

List 的重要接口用法介绍

1.创建和初始化List

在C++中,可以使用多种方式创建和初始化 list 容器:
在这里插入图片描述

list

  • (1)
    空容器构造函数(默认构造函数)构造一个没有元素的空容器。
  • (2)
    构造一个包含n个元素的容器。每个元素都初始化为val。
  • (3)
    构造一个包含与[first,last)范围相同数量元素的容器,每个元素以相同的顺序从该范围内的相应元素构造。
  • (4)
    构造了一个包含 x 中每个元素副本的容器,元素顺序与 x 中的顺序相同。
#include <iostream>
#include <list>int main() {// (1)创建一个空的liststd::list<int> mylist;// (2)创建一个有n个空间每个空间都存val的liststd::list<int> mylist2(5,10);// 使用初始化列表创建liststd::list<int> mylist3 = {1, 2, 3, 4, 5};// (3)创建一个包含一些整数的数组int arr[] = { 10, 20, 30, 40, 50 };// 创建一个指向数组的迭代器范围int* first = arr;int* last = arr + sizeof(arr) / sizeof(arr[0]);// 使用迭代器范围创建一个liststd::list<int> mylist3(first, last);// (4)使用另一个list初始化std::list<int> mylist4(mylist2);return 0;
}

2.插入元素

在这里插入图片描述

push_back

  • list 容器的末尾添加了一个新元素,位于其当前的最后一个元素之后。val 的内容被复制(或移动)到新元素中,就是尾插个节点
	std::list<int> lt;lt.push_back(1);lt.push_back(2);lt.push_back(3);

在这里插入图片描述

push_fornt

  • 就是头插个节点
    std::list<int> lt;lt.push_front(1);lt.push_front(2);lt.push_front(3);

在这里插入图片描述

insert

  • (1)
    就是指定位置的元素之前进行插入新元素
  • (2)
    选择插入位置进行单个或者多个元素的插入
  • (3)
    进行[first,last)范围的插入
	std::list<int> mylist1 = { 1, 2, 3, 4, 5 }; // 初始化列表std::list<int>::iterator it1 = mylist1.begin(); // 获取迭代器// 在迭代器 it 指向的位置之前插入值为 10 的元素it1 = mylist1.insert(it1, 10);//(1)std::list<int> mylist2 = { 1, 2, 3, 4, 5 }; // 初始化列表std::list<int>::iterator it2 = mylist2.begin(); // 获取迭代器// 在迭代器 it 指向的位置之前插入 2 个值为 10 的元素mylist2.insert(it2, 2, 10);//(2)std::list<int> mylist3 = { 1, 2, 3, 4, 5 }; // 初始化列表std::vector<int> myvector = { 6, 7, 8 }; // 初始化向量std::list<int>::iterator it3 = mylist3.begin(); // 获取迭代器// 在迭代器 it 指向的位置之前插入 myvector 中的元素mylist3.insert(it3, myvector.begin(), myvector.end());//(3)

删除元素

在这里插入图片描述

pop_back

  • 删除末尾元素

在这里插入图片描述


pop_font

  • 删除开头元素

在这里插入图片描述


clear

  • 清空整个list
mylist.pop_back(); 
mylist.pop_front(); 
mylist.clear();  

在这里插入图片描述

erase

  • (1)
    接受一个迭代器 position ,它指向你想要删除的单个元素。这个元素会被从容器中移除,容器的大小会减一。返回值是一个迭代器,指向被删除元素之后的元素,如果 position 是容器中的最后一个元素,则返回 end() 。
  • (2)
    接受一对迭代器 first 和 last ,它们定义了一个范围。这个范围内的所有元素都会被从容器中移除。返回值是一个迭代器,指向被删除范围之后的第一个元素,如果 last 是容器中的最后一个元素,则返回 end() 。
	std::list<int> mylist1 = { 1, 2, 3, 4, 5 };std::list<int>::iterator it = mylist1.begin();// 删除迭代器 it 指向的元素it = mylist1.erase(it);//(1)std::list<int> mylist2 = { 1, 2, 3, 4, 5 };std::list<int>::iterator it1 = mylist2.begin();std::list<int>::iterator it2 = mylist2.begin();std::advance(it2, 3); // 移动 it2 到第四个元素// 删除从 it1 到 it2 范围内的元素it1 = mylist2.erase(it1, it2);//(2)

遍历List

迭代器遍历

for (std::list<int>::iterator it = mylist.begin(); it != mylist.end(); ++it) {std::cout << *it << " ";
}

范围for遍历

	list<int> lt = { 1, 2, 3, 4, 5, 6, 7, 8, 9 };for (auto e : lt){cout << e << " ";}cout << endl;

排序List

在这里插入图片描述

sort

  • (1)
    升序排序
  • (2)
    降序排序(得需要仿函数)
	list<int> lt = { 1, 3, 2, 5, 4, 9, 8, 7, 6 };//排序升序lt.sort();lt.sort(greater<int>());//降序,greater<int>()//仿函数

反转List

在这里插入图片描述

reverse

  • 反转或者说逆置list
mylist.reverse(); // 反转list中的元素顺序

转移List

在这里插入图片描述

splice

  • 注意:
    转移之后lt3直接转移走,lt3里面什么都没有了就
  • 吐槽:
    虽说起名字叫splice但是还是感觉transfers更贴近一点
	//splice (transfers) 转移list<int> lt2;lt2.push_back(1);lt2.push_back(3);lt2.push_back(5);lt2.push_back(4);lt2.push_back(2);for (auto e : lt2){cout << e << " ";}cout << endl;lt2.splice(lt2.end(), lt2, lt2.begin());for (auto e : lt2){cout << e << " ";}cout << endl;list<int> lt3;lt3.push_back(1);lt3.push_back(2);lt3.push_back(3);for (auto e : lt3){cout << e << " ";}cout << endl;lt2.splice(lt2.end(), lt3);for (auto e : lt2){cout << e << " ";}cout << endl;//转移之后lt3直接转移走,lt3里面什么都没有了就for (auto e : lt3){cout << e << " ";}cout << endl;

去重

在这里插入图片描述

unique

  • (1)
    去除list中重复的元素
  • (2)
    接受一个二元谓词( BinaryPredicate ),用于定义两个元素是否相等。默认情况下,如果两个元素相等(即它们的值相同),则认为它们是重复的。unique 函数会遍历列表,并移除所有相邻的重复元素,只保留第一个元素。新列表的大小会减小,但元素的顺序不会改变。
  • 注意:
    在用去重函数之前一定要进行排序
	//去重 unique(得先排序)lt.unique();//(1)//(2)#include <iostream>#include <list>#include <algorithm> // for std::equal_tobool is_equal(const int& a, const int& b) {return a == b;}int main() {std::list<int> mylist = {1, 2, 2, 3, 3, 3, 4, 4, 4, 4};// 使用自定义的二元谓词来定义相等性mylist.unique(is_equal);// 打印列表以查看结果for (int& elem : mylist) {std::cout << elem << " ";}std::cout << std::endl;return 0;
}

合并

在这里插入图片描述

merge

  • 用于合并两个有序序列。 std::merge 函数定义在 头文件中,它接受两个有序序列的迭代器范围,并将它们合并成一个有序序列。
#include <iostream>
#include <list>
#include <algorithm> // for std::merge
#include <iterator>   // for std::ostream_iteratorint main() {std::list<int> list1 = {1, 3, 5, 7};std::list<int> list2 = {2, 4, 6, 8};std::list<int> merged_list;// 使用 std::merge 合并两个有序列表std::merge(list1.begin(), list1.end(),list2.begin(), list2.end(),std::back_inserter(merged_list));// 打印合并后的列表std::copy(merged_list.begin(), merged_list.end(),std::ostream_iterator<int>(std::cout, " "));std::cout << std::endl;return 0;
}

总结

list 是C++标准模板库中非常强大的容器之一,它提供了快速的插入和删除操作,并且支持双向迭代。通过本文的介绍,你应该对 list 的基本用法有了一个大致的了解。在实际编程中,根据具体需求选择合适的容器是非常重要的,list 适合于那些需要频繁插入和删除元素的场景。

希望这篇文章能够帮助你更好地理解和使用C++中的 list 容器。如果你有任何问题或需要进一步的解释,请随时在评论区留言。

小卷王们,用你们勤劳的小手给我点点赞,蟹蟹

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/589244.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【JVM】如何定位、解决内存泄漏和溢出

目录 1.概述 2.堆溢出、内存泄定位及解决办法 2.1.示例代码 2.2.抓堆快照 2.3.分析堆快照 1.概述 常见的几种JVM内存溢出的场景如下&#xff1a; Java堆溢出&#xff1a; 错误信息: java.lang.OutOfMemoryError: Java heap space 原因&#xff1a;Java对象实例在运行时持…

这里有份百度Create大会超长剧透,请查收!

作者简介&#xff1a; 辭七七&#xff0c;目前大二&#xff0c;正在学习C/C&#xff0c;Java&#xff0c;Python等 作者主页&#xff1a; 七七的个人主页 文章收录专栏&#xff1a; 七七的闲谈 欢迎大家点赞 &#x1f44d; 收藏 ⭐ 加关注哦&#xff01;&#x1f496;&#x1f…

【C++STL详解(二)】——string类模拟实现

目录 前言 一、接口总览 二、默认成员函数 1.构造函数 2.拷贝构造 写法一&#xff1a;传统写法 写法二&#xff1a;现代写法&#xff08;复用构造函数&#xff09; 3.赋值构造 写法一&#xff1a;传统写法 写法二&#xff1a;现代写法(复用拷贝构造) 4.析构函数 三、…

【flutter封装图片/视频选择控件】

引入库 wechat_assets_picker: ^6.0.5 、video_player: ^2.5.1 # 视频播放、 flutter_screenutil: ^5.7.0 import dart:async; import dart:io; import package:generated/l10n.dart; import package:jade/configs/PathConfig.dart; import package:jade/customWidget/addImag…

洛谷B3735题解

题目描述 圣诞树共有 n 层&#xff0c;从上向下数第 1 层有 1 个星星、第 2 层有 2 个星星、以此类推&#xff0c;排列成下图所示的形状。 星星和星星之间用绳子连接。第 1,2,⋯,n−1 层的每个星星都向下一层最近的两个星星连一段绳子&#xff0c;最后一层的相邻星星之间连一段…

算法知识点汇总

知识点 1. 求二进制中1的个数 int get_count(int x)//返回x的二进制有多少个1 int get_count(int x) {int res 0;while (x){res ;x - x & -x;}return res; }2. 建树&#xff0c;和树的DFS 记得初始化头节点 const int N 1e5 10, M N * 2; int h[N], e[M], ne[M], id…

RD55UP06-V 三菱iQ-R系列C语言功能模块

RD55UP06-V 三菱iQ-R系列C语言功能模块 RD55UP06-V用户手册&#xff0c;RD55UP06-V功能&#xff0c;RD55UP06-V系统配置 RD55UP06-V参数规格&#xff1a;10BASE-T/100BASE-TX/1000BASE-T 1通道&#xff1b;字节存储次序格式小端模式; 可使用SD存储卡插槽&#xff1b;工作RAM 1…

争光树脂邀您到场参观2024年第13届生物发酵展

参展企业介绍 宁波争光树脂有限公司成立于2006年11月&#xff0c;是浙江争光实业股份有限公司的全资子公司&#xff0c;公司专业生产离子交换树脂&#xff0c;产品的应用领域主要涉及电厂、核能、石油、化工、轻工、医药、食品、饮料、冶金、环保、生物等领域&#xff0c;年生…

AGV无人驾驶跨境运输新模式引领未来物流

agv AGV即“自动导引运输车”&#xff0c;这一概念起源于欧美&#xff0c;在欧美及日韩市场的发展比较成熟&#xff0c;于上世纪末被引入国内。这种自动导引运输车可以广泛应用于汽车、化工、医药以及食品饮料等制造业场景&#xff0c;以及机场、码头等仓储物流行业场景&#x…

JavaScript中什么叫深拷贝?

在 JavaScript 中&#xff0c;深拷贝指的是创建一个新的对象&#xff0c;这个新的对象与原始对象完全独立&#xff0c;没有任何共享的属性或者数据&#xff0c;它们不共享同一块内存地址。深拷贝会复制原始对象的所有属性和嵌套对象的所有属性&#xff0c;包括嵌套对象中的属性…

【深度学习】图像自然语言描述生成

案例 6&#xff1a;图像自然语言描述生成&#xff08;让计算机“看图说话”&#xff09; 相关知识点&#xff1a;RNN、Attention 机制、图像和文本数据的处理 1 任务目标 1.1 任务和数据简介 ​ 本次案例将使用深度学习技术来完成图像自然语言描述生成任务&#xff0c;输入…

物联网实战--入门篇之(四)嵌入式-UART驱动

目录 一、串口简介 二、串口驱动设计 三、串口发送 四、串口接收处理 五、PM2.5数据接收处理 六、printf重定义 七、总结 一、串口简介 串口在单片机的开发中属于非常常用的外设&#xff0c;最基本的都会预留一个调试串口用来输出调试信息&#xff0c;串口时序这里就不谈…