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

news2024/11/20 10:28:32

文章目录

  • 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)创建一个空的list
    std::list<int> mylist;
	
	// (2)创建一个有n个空间每个空间都存val的list
	std::list<int> mylist2(5,10);
	// 使用初始化列表创建list
    std::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]);
	// 使用迭代器范围创建一个list
	std::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_to

	bool 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_iterator

int 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.coloradmin.cn/o/1566845.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

wordpress课程项目主题电脑版+手机版自适应

这款主题适合做资源、课程、素材等&#xff0c;演示站&#xff1a;点击查看

​网络socket编程(二)——面向流的TCP编程及测试(SocketTool)、Wireshark软件使用

目录 一、书接上回&#xff08;select()函数使用注意事项&#xff09; 二、面向流(TCP)的socket编程 2.1 TCP服务端编程和测试 2.1.1 TCP服务器原理流程图 2.1.2 TCP服务端编程实战 2.1.3 测试 2.2 TCP客户端编程和测试 三、Wireshark抓包软件的使用 3.1 Wireshark是什…

fastlio2 给 interactive-slam 保存每帧的点云和每帧的里程计为单独的文件做后端回环优化和手动回环优化

为了给 interactive-slam 提供数据做后端回环优化和手动回环优化,需要保存每帧的点云和每帧的里程计为单独的文件,并且需要保存的名字为ros时间戳。 效果很好,比我自己写的手动回环模块好用 // This is an advanced implementation of the algorithm described in the // fo…

Jmeter各组件超详细介绍

1、JMeter和Loadrunner的区别&#xff1f; 2、JMeter如何开发脚本的&#xff1f;强化脚本的技术&#xff1f; 代理服务器录制脚本&#xff0c;Fiddler录制脚本&#xff0c;Badboy录制脚本&#xff0c;根据API&#xff0c;手写脚本&#xff0c;根据抓包&#xff0c;手写脚本。 …

Twitter Api查询用户粉丝列表

如果大家为了获取实现方式代码的话可能要让大家失望了&#xff0c;这边文章主要是为了节省大家开发时间&#xff0c;少点坑。https://api.twitter.com/2/users/:id/followers &#xff0c;这个接口很熟悉吧&#xff0c;他是推特提供的获取用户关注者&#xff08;粉丝&#xff0…

目标跟踪——行人车辆数据集

一、重要性及意义 首先&#xff0c;目标跟踪对于个人和组织的目标实现至关重要。无论是个人职业发展、企业业务增长还是政府的社会发展&#xff0c;目标跟踪都能够帮助我们明确目标&#xff0c;并将其分解为可行的步骤和时间表。这有助于我们保持动力和专注&#xff0c;提高效…

CNAS软件测试公司有什么好处?如何选择靠谱的软件测试公司?

CNAS认可是中国合格评定国家认可委员会的英文缩写&#xff0c;由国家认证认可监督管理委员会批准设立并授权的国家认可机构&#xff0c;统一负责对认证机构、实验室和检验机构等相关机构的认可工作。 在软件测试行业&#xff0c;CNAS认可具有重要意义。它标志着一个软件测试公…

Java并发编程基础面试题详细总结

1. 什么是线程和进程? 1.1 何为进程? 进程是程序的一次执行过程&#xff0c;是系统运行程序的基本单位&#xff0c;因此进程是动态的。系统运行一个程序即是一个进程从创建&#xff0c;运行到消亡的过程。 在 Java 中&#xff0c;当我们启动 main 函数时其实就是启动了一个…

Failed to resolve import “Home/components/HomeNew.vue“. Does the file exist?

错误信息 [plugin:vite:import-analysis] Failed to resolve import "/apis/home.js" from "src/views/Home/components/HomeNew.vue". Does the file exist? 错误原因 路径错误 解决方法

[C#]使用OpencvSharp去除面积较小的连通域

【C介绍】 关于opencv实现有比较好的算法&#xff0c;可以参考这个博客OpenCV去除面积较小的连通域_c#opencv 筛选小面积区域-CSDN博客 但是没有对应opencvsharp实现同类算法&#xff0c;为了照顾懂C#编程同学们&#xff0c;因此将 去除面积较小的连通域算法转成C#代码。 方…

Open CASCADE学习|刚体( TopoDS_Shape)按某种轨迹运动,停在指定位置上

今天实现如下功能&#xff1a;刚体做做螺旋运动&#xff0c;轨迹已知&#xff0c;求刚体在每个位置上的所占据的空间&#xff0c;就是把刚体从初始位置变换到该位置。 这里的刚体是一个砂轮截面&#xff0c;螺旋运动轨迹由B样条曲线拟合&#xff0c;通过Frenet标架确定运动轨迹…

Spring Boot集成AJ-Captcha实现滑动验证码功能

1.AJ-Captcha介绍 行为验证码 采用嵌入式集成方式&#xff0c;接入方便&#xff0c;安全&#xff0c;高效。抛弃了传统字符型验证码展示-填写字符-比对答案的流程&#xff0c;采用验证码展示-采集用户行为-分析用户行为流程&#xff0c;用户只需要产生指定的行为轨迹&#xff0…

WPS二次开发系列:如何获取应用签名SHA256值

在申请WPS SDK授权版时候需要开发者提供应用包名和签名&#xff0c;应用包名好说&#xff0c;那如何生成符合WPS要求的应用签名&#xff08;SHA256)呢&#xff0c;经笔者亲测&#xff0c;有如下两种方式可以实现获取第三方应用签名值&#xff08;SHA256&#xff09; 1. 方法一&…

springboot对接minio的webhook全过程

前言 近日需要将minio的apache2.0版本给用起来&#xff0c;顺便要完善一下原有的文件上传管理系统&#xff0c;其中很重要的一点是&#xff0c;在原有客户端直传的基础上&#xff0c;再添加 minio 的上传回调给服务端做后续处理。 本文重点在于&#xff0c;介绍整个minio与spr…

SpringCloud学习(1)-consul

consul下载安装及使用 1.consul简介 Consul是一种开源的、分布式的服务发现和配置管理工具&#xff0c;能够帮助开发人员构建和管理现代化的分布式系统。它提供了一套完整的功能&#xff0c;包括服务注册与发现、健康检查、KV存储、多数据中心支持等&#xff0c;可以帮助开发人…

Jenkins--任务详解

一、任务类型 Jenkins的主要功能的实现是由执行任务去完成的&#xff0c;常用的任务类型主要有以下三种&#xff1a; 自由风格任务(Free Style Project): 这是Jenkins中最常用的任务类型&#xff0c;允许你自定义各种构建步骤和配置选项&#xff0c;如源码管理、构建触发器、…

vue3+echarts:echarts地图打点显示的样式

colorStops是打点的颜色和呼吸灯、label为show是打点是否显示数据、rich里cnNum是自定义的过滤模板用来改写显示数据的样式 series: [{type: "effectScatter",coordinateSystem: "geo",rippleEffect: {brushType: "stroke",},showEffectOn: &quo…

Redis的值有5种数据结构,不同数据结构的使用场景是什么?

文章目录 字符串缓存计数共享Session限速 哈希缓存 列表消息队列文章列表栈队列有限集合 集合标签抽奖社交需求 有序集合排行榜系统 字符串 缓存 &#xff08;1&#xff09;使用原生字符类型缓存 优点&#xff1a;简单直观&#xff0c;每个属性都支持更新操作 缺点&#xff1…

大话设计模式之状态模式

状态模式是一种行为设计模式&#xff0c;它允许对象在其内部状态发生变化时改变其行为。在状态模式中&#xff0c;对象将其行为委托给当前状态对象&#xff0c;从而在不同的状态下执行不同的行为&#xff0c;而不必在对象自身的代码中包含大量的条件语句。 通常&#xff0c;状…

WE博客代码系统

WE博客代码系统 说明文档 运行前附加数据库.mdf&#xff08;或sql生成数据库&#xff09; 主要技术&#xff1a; 基于asp.net mvc架构和sql server数据库&#xff0c;并采用EF实体模型开发。 三层架构并采用EF实体模型开发 功能模块&#xff1a; WE博客代码系统 WE博客代码系…