深入解析【C++ list 容器】:高效数据管理的秘密武器

 

目录

1. list 的介绍及使用

1.1 list 的介绍

知识点:

小李的理解:

1.2 list 的使用

1.2.1 list 的构造

知识点:

小李的理解:

代码示例:

1.2.2 list 迭代器的使用

知识点:

小李的理解:

代码示例:

1.2.3 list 的容量

知识点:

小李的理解:

代码示例:

1.2.4 list 的元素访问

知识点:

小李的理解:

代码示例:

1.2.5 list 的修改操作

知识点:

小李的理解:

代码示例:

1.2.6 list 迭代器失效

知识点:

小李的理解:

代码示例:

2. list 的模拟实现

知识点:

小李的理解:

代码示例:

3. list 与 vector 的对比

知识点:

小李的理解:

 总结


专栏:C++学习笔记 

上一卷:【C++ 】-vector:新时代动态数组的革新与未来

C++ 中的 list 是一个强大的容器,特别适用于需要频繁插入和删除元素的场景。本文将详细介绍 list 容器,包括其介绍、使用方法、实现原理以及与 vector 容器的对比。

1. list 的介绍及使用

1.1 list 的介绍

知识点:

list 是一种序列式容器,底层实现为双向链表。双向链表中的每个元素存储在独立的节点中,节点通过指针互相连接,可以在常数时间内在任意位置进行插入和删除操作。与 forward_list 的单链表不同,list 支持双向迭代。与其他序列式容器(如 arrayvectordeque)相比,list 在插入和删除操作方面表现更优,但不支持随机访问。

小李的理解:

list 就像一条双向街道上的车队,每辆车(节点)都有前后两个链接,指向前后两辆车。你可以轻松地在任何地方插入或删除一辆车,而不需要移动其他车。但是,如果你想找到某辆车,就需要从头或尾开始,一辆辆查找,比较费时。

1.2 list 的使用

1.2.1 list 的构造

知识点:

list 提供多种构造方法,包括创建空的 list,创建包含多个相同元素的 list,使用区间构造 list 以及拷贝构造等。

小李的理解:

构造 list 就像创建不同类型的车队,你可以创建一个空车队,或者一个全是同样车的车队,还可以用现有的车队来创建新的车队。

代码示例:
#include <list>
#include <iostream>int main() {// 创建一个空的 liststd::list<int> empty_list;// 创建一个包含 5 个值为 10 的元素的 liststd::list<int> filled_list(5, 10);// 使用区间构造 listint arr[] = {1, 2, 3, 4, 5};std::list<int> range_list(arr, arr + 5);// 打印 filled_list 的元素for(int n : filled_list) {std::cout << n << " ";}std::cout << std::endl;return 0;
}

这个程序首先创建一个空的 list,然后创建一个包含 5 个值为 10 的 list,接着用数组中的元素构造一个 list。最后打印 filled_list 的元素,显示 5 个 10。

1.2.2 list 迭代器的使用

知识点:

list 的迭代器类似于指针,指向 list 中的节点。你可以使用迭代器遍历 list,访问或修改元素。

小李的理解:

迭代器就像你走在车队中的一个人,你可以走到每辆车旁边查看里面的东西,或者往回走查看后面的车。

代码示例:
#include <list>
#include <iostream>int main() {std::list<int> my_list = {1, 2, 3, 4, 5};// 正向迭代for (auto it = my_list.begin(); it != my_list.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;// 反向迭代for (auto rit = my_list.rbegin(); rit != my_list.rend(); ++rit) {std::cout << *rit << " ";}std::cout << std::endl;return 0;
}

这个程序创建一个包含 5 个整数的 list。正向迭代器遍历 list,从头到尾打印元素,反向迭代器遍历 list,从尾到头打印元素。

1.2.3 list 的容量

知识点:

你可以使用 empty() 函数检查 list 是否为空,使用 size() 函数获取 list 中的元素个数。

小李的理解:

这就像你检查车队中是否有车,以及数一数车队中有多少辆车。

代码示例:
#include <list>
#include <iostream>int main() {std::list<int> my_list = {1, 2, 3, 4, 5};// 检查是否为空if (my_list.empty()) {std::cout << "List is empty" << std::endl;} else {std::cout << "List is not empty" << std::endl;}// 获取大小std::cout << "List size: " << my_list.size() << std::endl;return 0;
}

 

程序首先检查 list 是否为空,显然 my_list 不为空,然后打印 list 的大小,显示有 5 个元素。

1.2.4 list 的元素访问

知识点:

你可以使用 front() 函数访问 list 的第一个元素,使用 back() 函数访问 list 的最后一个元素。

小李的理解:

这就像你想看看车队的第一辆车和最后一辆车里装了什么东西。

代码示例:
#include <list>
#include <iostream>int main() {std::list<int> my_list = {1, 2, 3, 4, 5};// 访问第一个和最后一个元素std::cout << "First element: " << my_list.front() << std::endl;std::cout << "Last element: " << my_list.back() << std::endl;return 0;
}

程序分别打印 list 的第一个和最后一个元素,显示 1 和 5。

1.2.5 list 的修改操作

知识点:

list 提供丰富的修改操作,包括在头部和尾部插入和删除元素,插入和删除特定位置的元素,交换两个 list 的内容以及清空 list

小李的理解:

这就像你可以在车队的任何位置加车或减车,甚至可以交换两队车里的车,或者把整个车队清空。

代码示例:

 

#include <list>
#include <iostream>int main() {std::list<int> my_list = {1, 2, 3, 4, 5};// 插入和删除操作my_list.push_front(0);  // 在前面插入 0my_list.push_back(6);   // 在后面插入 6my_list.pop_front();    // 删除第一个元素my_list.pop_back();     // 删除最后一个元素// 插入和删除特定位置的元素auto it = my_list.begin();++it;  // 指向第二个元素my_list.insert(it, 100);  // 在第二个位置插入 100it = my_list.begin();++it;my_list.erase(it);  // 删除第二个位置的元素// 交换和清空 liststd::list<int> another_list = {10, 20, 30};my_list.swap(another_list);my_list.clear();return 0;
}

以下是程序在各步之后的状态:

  1. my_list 初始为 {1, 2, 3, 4, 5}
  2. 插入 0 和 6 后为 {0, 1, 2, 3, 4, 5, 6}
  3. 删除第一个和最后一个元素后为 {1, 2, 3, 4, 5}
  4. 在第二个位置插入 100 后为 {1, 100, 2, 3, 4, 5}
  5. 删除第二个位置的元素后为 {1, 2, 3, 4, 5}
  6. another_list 交换后 my_list{10, 20, 30}
  7. 清空后 my_list 为空

1.2.6 list 迭代器失效

知识点:

list 中,插入操作不会导致迭代器失效,删除操作会使指向被删除节点的迭代器失效。

小李的理解:

就像你从车队中移走一辆车时,那个位置的指示牌(迭代器)也被移走了,但其他位置的指示牌不受影响。

代码示例:
#include <list>
#include <iostream>int main() {std::list<int> my_list = {1, 2, 3, 4, 5};auto it = my_list.begin();while (it != my_list.end()) {it = my_list.erase(it);  // 删除元素后,更新迭代器}return 0;
}

这个程序将删除 list 中的所有元素,最后 my_list 为空。

每次删除一个元素后,迭代器指向下一个元素,直到 list 清空。

2. list 的模拟实现

知识点:

实现一个简化版本的 list,需要理解其底层结构和接口的含义。以下是一个简化的 list 实现示例:

小李的理解:

模拟实现 list 就像你自己动手造一辆汽车,你需要理解汽车的每个部件和它们如何协同工作。

代码示例:

#include <iostream>template<typename T>
class Node {
public:T data;Node* prev;Node* next;Node(T val) : data(val), prev(nullptr), next(nullptr) {}
};template<typename T>
class List {
private:Node<T>* head;Node<T>* tail;public:List() : head(nullptr), tail(nullptr) {}void push_back(T val) {Node<T>* newNode = new Node<T>(val);if (!tail) {head = tail = newNode;} else {tail->next = newNode;newNode->prev = tail;tail = newNode;}}void print() {Node<T>* temp = head;while (temp) {std::cout << temp->data << " ";temp = temp->next;}std::cout << std::endl;}~List() {Node<T>* temp;while (head) {temp = head;head = head->next;delete temp;}}
};int main() {List<int> my_list;my_list.push_back(1);my_list.push_back(2);my_list.push_back(3);my_list.print();return 0;
}

这个程序手动实现了一个简单的 list,并添加了 3 个元素。最终打印出 list 中的所有元素。

3. listvector 的对比

知识点:

  • 底层结构

    • vector:动态顺序表,连续内存空间。
    • list:双向链表,不连续。
  • 随机访问

    • vector:支持,效率为 O(1)。
    • list:不支持,效率为 O(N)。
  • 插入和删除

    • vector:效率低,时间复杂度为 O(N)。
    • list:效率高,时间复杂度为 O(1)。
  • 空间利用率

    • vector:高,连续空间,缓存利用率高。
    • list:低,节点动态分配,容易造成内存碎片。
  • 迭代器

    • vector:原生指针。
    • list:封装的节点指针。
  • 迭代器失效

    • vector:插入时可能失效,删除时当前迭代器失效。
    • list:插入不会失效,删除时当前迭代器失效。
  • 使用场景

    • vector:高效存储,随机访问。
    • list:频繁插入和删除操作。

小李的理解:

vector 就像一块连续的停车场,每辆车(元素)都紧挨着,如果你要在中间插入或删除一辆车,就需要挪动很多车。而 list 就像一列火车,每节车厢(元素)独立,可以随意插入或移除车厢,但要找到某个特定车厢就得一节一节地找。

 总结

C++ 中的 list 容器是一个基于双向链表的序列式容器,适用于需要频繁插入和删除操作的场景,但不支持随机访问。list 提供了多种构造方法和丰富的操作接口,包括插入、删除、访问等。与 vector 相比,list 在插入和删除操作上更高效,但在随机访问和空间利用率上较差。

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

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

相关文章

基于STM32的气压检测报警proteus仿真设计(仿真+程序+设计报告+讲解视频)

基于STM32的气压检测报警proteus仿真设计 1.主要功能2.仿真3. 程序4. 设计报告5. 资料清单&下载链接资料下载链接&#xff1a; 基于STM32的气压检测报警proteus仿真设计(仿真程序设计报告讲解视频&#xff09; 仿真图proteus 8.9 程序编译器&#xff1a;keil 5 编程语言…

【Linux】多线程_1

文章目录 九、多线程1. 线程概念2. 线程的控制 未完待续 九、多线程 1. 线程概念 我们知道&#xff1a;进程 内核数据结构 进程代码和数据 。那什么是线程呢&#xff1f;线程是进程内部的一个执行分支。一个进程内部可以有多个执行流&#xff08;内核数据结构&#xff09;&…

如何用Vue3和Plotly.js创建交互式平行坐标图

本文由ScriptEcho平台提供技术支持 项目地址&#xff1a;传送门 Vue.js 中使用 Plotly.js 创建平行坐标图 应用场景介绍 平行坐标图是一种可视化高维数据集的强大技术。它可以帮助我们探索不同维度之间的关系&#xff0c;并识别模式和异常值。在 Vue.js 应用程序中&#xf…

telnet在windows和linux上的使用方法

telnet在windows上使用 ‘telnet’ 不是内部或外部命令&#xff0c;也不是可运行的程序或批处理文件。 windows上有自带的telnet工具的&#xff0c;这只是没有安装添加进来而已。 处理 方法&#xff1a; 打开控制面板-点击程序与功能 进到程序与功能界面&#xff0c;点击启用或…

S32V234平台开发(一)快速使用

快速使用 准备供电复位选择串口通信启动选择显示登陆系统 准备供电 s32v234可以使用两种电源供电 一种是左边电源端子&#xff0c;一种是右边电源适配器(12V 3A) 注意:不要同时使用两种电源同时供电 复位选择 Pressing POR RESET pulls active low EXT_POR signal on S32V2…

deepstream段错误

&#x1f610; 错误&#xff1a; 探针中由于使用了pyds.get_nvds_buf_surface(hash(gst_buffer), frame_meta.batch_id)导致的段错误&#xff08;segmentation fault&#xff09;。 解决方式&#xff1a;

职场中的3个误区,你踩坑了吗?

1、个人发展比工资待遇更重要 这句话也不能说是完全错的&#xff0c;但是你要明白能给你提供发展空间的公司&#xff0c;待遇也不会差到哪里去&#xff0c;而且随着你个人能力的提升&#xff0c;发展也会越来越好&#xff0c;你的待遇也自然水涨船高&#xff0c;这个道理其实大…

为什么选择虚拟艺术设计展览?艺术家应知的五个关键好处

随着技术的进步&#xff0c;虚拟艺术设计展览成为了展示艺术作品的重要途径。它不仅为艺术家们提供了新的展示平台&#xff0c;还打破了传统展览的局限。 1、扩大观众范围&#xff1a;打破地理限制 虚拟艺术设计展览能够打破地理限制&#xff0c;使得全球观众可以随时随地访问…

go GMP 模型,为什么要有 P?

GM 模型 在 Go1.1 之前 Go 的调度模型其实就是 GM 模型&#xff0c;也就是没有 P。 static void schedule(G *gp) {...schedlock();if(gp ! nil) {...switch(gp->status){case Grunnable:case Gdead:// Shouldnt have been running!runtimethrow("bad gp->status …

春秋杯 snack入土为安的第二天

不嘻嘻&#xff0c;签到题做了两天&#xff0c;先用pyinstxtractor.py&#xff08;找最新版本。。红温&#xff09;把exe转化为pyc&#xff0c;用在线反编译pycdc来反编译&#xff0c;最后的key在一个文件夹里key.pyc切记用python3.3版本&#xff08;红温&#xff09;。 # 假设…

同步时钟系统支持多种校时方式

在当今数字化、信息化高速发展的时代&#xff0c;时间的准确性和同步性变得至关重要。无论是金融交易、通信网络、交通运输&#xff0c;还是工业生产、科学研究等领域&#xff0c;都离不开一个精确且同步的时钟系统。而同步时钟系统之所以能够在众多领域发挥关键作用&#xff0…

手把手和你一起从0搭建一个vite+Vue3+element-plus的后台管理系统!

首选需要node环境。你可以参考&#xff1a;如何创建一个vue的新项目&#xff0c;用命令行的方式创建.下载node.js-CSDN博客 所需要的内容官网&#xff1a; https://vitejs.dev/ 安装 | Element Plus Vue Router | Vue.js 的官方路由 axios中文网|axios API 中文文档 | axio…

【Linux进阶】文件系统2——MBR和GPT

1.磁盘的分区 因为如果你的磁盘被划分成两个分区&#xff0c;那么每个分区的设备文件名是什么&#xff1f; 在了解这个问题之前&#xff0c;我们先来复习一下磁盘的组成&#xff0c;因为现今磁盘的划分与它物理的组成很有关系。 我们谈过磁盘主要由碟片、机械手臂、磁头与主轴马…

1125 子串与子列

solution #include<iostream> #include<string> using namespace std; int main(){int l, r, len 10000, n;string s, p;cin >> s >> p;for(int i 0; i < s.size() - p.size(); i){//剩下子串长度不足p.siez()时&#xff0c;不可能再出现子列pif…

项目收获总结--MyBatis的知识收获

一、概述 最近几天公司项目开发上线完成&#xff0c;做个收获总结吧~ 今天记录MyBatis的收获和提升。 二、获取自动生成的(主)键值 insert 方法总是返回一个 int 值 &#xff0c;这个值代表的是插入的行数。若表的主键id采用自增长策略&#xff0c;自动生成的键值在 insert…

飞跃边界,尽在掌握 —— Jump Desktop 8 for Mac,远程工作新体验!

Jump Desktop 8 for Mac 是一款强大的远程桌面控制软件&#xff0c;专为追求高效工作与生活平衡的用户设计。它允许您轻松地从Mac设备上远程访问和控制另一台电脑或服务器&#xff0c;无论是跨房间、跨城市还是跨国界&#xff0c;都能实现无缝连接&#xff0c;仿佛操作就在眼前…

使用命令行修改Ubuntu 24.04的网络设置

Ubuntu里&#xff0c;使用命令行下修改IP地址&#xff0c;网上有很多方案&#xff0c;我最终觉得这个方案&#xff08;使用Netplan&#xff09;最好&#xff0c;最根本&#xff0c;记录下来备查 1.使用命令ip link show 查看Ubuntu上可以使用的网络接口名称 2.查找Netplan的配…

Tomcat部署以及优化

1、tomcat tomcat是用java代码的程序&#xff0c;运行的是java的web服务器 tomcat和php一样都是用来处理动态页面&#xff0c;tomcat也可以做为web应用服务器&#xff0c;开源的 php处理.php为结尾的 tomcat 处理.jsp nginx 处理 .html 2、tomcat的特点和功能 1、servle…

springcloud分布式架构网上商城 LW +PPT+源码+讲解

3系统分析 3.1可行性分析 在开发系统之前要进行系统可行性分析&#xff0c;目的是在用最简单的方法去解决最大的问题&#xff0c;程序一旦开发出来满足了用户的需要&#xff0c;所带来的利益也很多。下面我们将从技术、操作、经济等方面来选择这个系统最终是否开发。 3.1.1技术…

EasyCVR视频汇聚平台:存储系统怎么选?分布式存储vs.集中式存储的区别在哪?

在当今的数字化时代&#xff0c;安防监控已成为维护社会秩序和公共安全的重要手段。随着监控设备的普及和监控数据的不断增加&#xff0c;如何高效、安全地存储和管理这些视频数据&#xff0c;成为了安防行业面临的重要挑战。EasyCVR视频存储系统凭借其卓越的性能和灵活的架构&…