【C++】中Vector与List的比较

news2024/11/17 9:36:14

目录

1.List的介绍

2.List的使用         

3List和Vector的比较

———————————————————————————————————————————

下面让我们来了解一下List及List和Vector的区别

正文开始-->

1.List的介绍

  • 在 C++ 中,list是标准模板库(STL)中的一个容器。它是一个双向链表,与vector不同,list中的元素在内存中不是连续存储的。这使得list在插入和删除操作(特别是在序列中间进行操作)时具有较高的效率,因为不需要移动大量元素来腾出或填充空间。List的头文件是<list>,在使用list之前需要包含这个头文件。
  • 由于std::list是链表结构,元素在内存中不是像vector那样连续存储的。这意味着它不会像vector一样,因为内存重新分配而导致元素的大量复制。
  • 例如,当向一个vector中插入大量元素,导致其内存空间不足时,vector会重新分配一块更大的内存空间,并将原有的元素复制到新的空间中。而std::list则不存在这个问题,它可以很灵活地在链表的任何位置插入或删除节点,而不需要移动其他节点。

2.List的使用         

定义和初始化:

定义一个空的List:

#include <list>
int main() {
    std::list<int> myList;
    return 0;
}

 用已有元素初始化List:                                            

std::list<int> myList1 = {1, 2, 3};
std::list<int> myList2(myList1);

添加元素

在尾部添加元素用push_back()函数,头部添加用push_front()函数,例如:

//尾部添加元素
std::list<int> myList;
myList.push_back(1);
myList.push_back(2);
myList.push_back(3);


//头部添加元素
std::list<int> myList;
myList.push_front(3);
myList.push_front(2);
myList.push_front(1);

在指定位置插入元素:使用insert()函数。例如:在某个迭代器位置插入元素

std::list<int> myList = {2, 3};
auto it = myList.begin();
myList.insert(it, 1);

这里begin()是指向List第一个元素的迭代器,插入操作是在这个迭代器指向的元素之前插入新元素

访问元素

使用迭代器访问元素:List通常用迭代器来访问元素,例如:

std::list<int> myList = {1, 2, 3};
std::list<int>::iterator it;
for (it = myList.begin(); it!= myList.end(); ++it) 
{
    std::cout << *it << " ";
}

其中begin()是指向第一个元素的迭代器,end()是指向最后一个元素的迭代器

但是list中,不能像数组一样直接使用下标访问.因为List元素在内存中不是连续的,所以不能像vector一样直接使用[]操作符来进行访问元素.

删除元素

 删除头部元素和删除尾部元素:删除头部元素使用pop_front()函数,尾部元素使用pop_back()函数,例如

//删除头部元素
std::list<int> myList = {1, 2, 3};
myList.pop_front();
//删除尾部元素
std::list<int> myList = {1, 2, 3};
myList.pop_back();

删除指定位置的元素:使用erase()函数,例如,删除某个迭代器指向的元素:

std::list<int> myList = {1, 2, 3};
auto it = myList.begin();
++it;
myList.erase(it);

获取大小:可以使用size()函数来获取List中元素的个数:

std::list<int> myList = {1, 2, 3};
std::cout << "Size of the list: " << myList.size() << std::endl;

List的实际应用场景:

  • 频繁插入和删除操作的场景:例如,在一个文本编辑器中,用于存储文本行,当用户插入或删除行时,list的高效插入和删除特性就可以发挥作用。
  • 需要保持元素插入顺序的场景:比如记录用户操作的历史记录,每个操作作为一个元素插入到list中,按照操作的先后顺序存储。

3List和Vector的比较

内存布局

vector:

  • vector的元素在内存中是连续存储的。就像是一个数组,这种连续存储的方式使得它可以通过指针运算来实现快速的随机访问。例如,对于一个vector<int> v,如果知道第一个元素的地址(&v[0]),那么可以通过简单的偏移量计算来访问其他元素,如v[2]的地址就是&v[0]+ 2*sizeof(int)。
  • 由于内存连续,vector在存储大量元素时,如果需要重新分配内存(例如,当添加元素导致当前内存空间不足时),会把所有元素复制到新的内存空间中。这可能会导致性能开销,特别是在元素数量很大的情况下。

List

  • list是双向链表结构,元素在内存中不是连续存储的。每个元素(节点)都包含指向前一个节点和后一个节点的指针。这种存储方式使得它在插入和删除元素时不需要移动其他元素,只需要调整节点之间的指针关系。
  • 例如,在list中插入一个新元素,只需要修改新元素前后节点的指针,使其正确地链接到新元素即可。这种内存布局使得list在频繁插入和删除操作时比vector更具优势。

访问元素

vector

  • vector支持快速的随机访问。可以使用下标操作符[]或者at函数来访问元素。例如,对于vector<int> v = {1,2,3},访问v[1]的时间复杂度是,这意味着无论vector中有多少个元素,访问任意一个元素的时间基本是固定的。
  • 不过,使用[]操作符时要注意边界检查,因为如果下标超出范围,可能会导致程序出现未定义行为。at函数则会进行边界检查,当越界时会抛出std::out_of_range异常。

List

  • list不支持像vector那样高效的随机访问。如果要访问list中的一个特定元素,需要从链表的头部或者尾部开始遍历。例如,对于std::list<int> l = {1,2,3},如果要访问第二个元素,通常需要使用迭代器从头部开始遍历,时间复杂度是,其中是链表中元素的个数。
  • 虽然list也有迭代器可以用于遍历元素,但它的迭代器不能像vector的下标那样进行简单的算术运算来快速定位元素。

插入和删除操作

vector

  • 在vector的末尾插入元素是很快的,时间复杂度为。例如,使用push_back函数向vector的末尾添加元素,只需要在当前已分配的内存空间的末尾添加新元素即可。
  • 但是,在vector的中间或者开头插入元素就比较复杂。因为元素是连续存储的,插入一个元素可能需要将插入位置之后的所有元素都向后移动一个位置,以腾出空间给新元素.

list

  • list在任意位置插入和删除元素的时间复杂度都是。无论是在头部、中间还是尾部插入或删除元素,只需要调整相关节点的指针即可。例如,使用insert函数在list的指定位置插入元素,或者使用erase函数删除指定位置的元素,操作相对简单且高效。

今天就分享到这里了😁😁😁

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

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

相关文章

多模态大模型开启AI社交新纪元,Soul App创始人张璐团队亮相2024 GITEX GLOBAL

随着AI在全球范围内的加速发展和广泛应用,各行业纷纷在此领域发力。作为全球最大的科技盛会之一,2024年的GITEX GLOBAL将目光再次聚焦于人工智能的飞速发展,吸引了超过6700家来自各个领域的企业参与。在这样的背景下,Soul App作为国内较早将AI技术应用于社交领域的平台,首次亮相…

67页PDF |埃森哲_XX集团信息发展规划IT治理优化方案(限免下载)

一、前言 这份报告是埃森哲_XX集团信息发展规划IT治理优化方案&#xff0c;报告中详细阐述了XX集团如何优化IT治理结构以适应新的要求。报告还分析了集团管控模式的变化&#xff0c;提出了六大业务中心的差异化管控策略&#xff0c;并探讨了这些变化对IT治理模式的影响。报告进…

基于java Springboot高校失物招领平台

一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 数据…

[C++] 智能指针

文章目录 智能指针的使用原因及场景分析为什么需要智能指针&#xff1f;异常抛出导致的资源泄漏问题分析 智能指针与RAIIC常用智能指针 使用智能指针优化代码优化后的代码优化点分析 析构函数中的异常问题解决方法 RAII 和智能指针的设计思路详解什么是 RAII&#xff1f;RAII 的…

Git回到某个分支的某次提交

1.切换到需要操作的分支&#xff08;<branch-name>是分支名称&#xff09;。 命令如下&#xff1a; git checkout <branch-name> 2.获取代码的提交记录 。命令如下&#xff1a; git log 按q退出当前命令对话。 获取到某次提交或者合并的hash值&#xff08;下文…

掌握 Spring Boot 的最佳方法 – 学习路线图

在企业界&#xff0c;人们说“Java 永垂不朽&#xff01;”。但为什么呢&#xff1f;Java 仍然是开发企业应用程序的主要平台之一。大型公司使用企业应用程序来赚钱。这些应用程序具有高可靠性要求和庞大的代码库。根据Java开发人员生产力报告&#xff0c;62% 的受访开发人员使…

1. Django中的URL调度器 (项目创建与简单测试)

1. 创建 Django 项目 运行以下命令创建一个名为 blog_project 的 Django 项目&#xff1a; django-admin startproject blog_project2. 创建博客应用 Django 中&#xff0c;项目可以包含多个应用。创建一个名为 blog 的应用&#xff1a; cd blog_project python manage.py …

frp内网穿透介绍安装教程

文章目录 前言一、安装二、测试使用总结 前言 内网穿透&#xff08;Port Forwarding&#xff09;是将公网上的IP地址映射到内部网络中的一台计算机的某个端口上&#xff0c;以便外部网络可以访问该计算机中运行的应用程序。内网穿透技术可以通过一些开源工具来实现&#xff0c…

主界面获取个人信息客户端方

主界面获取个人信息客户端方 前言 上一集我们完成了websocket身份验证的内容&#xff0c;那么这一集开始我们将要配合MockServer来完成主界面获取个人信息的内容。 需求分析 我们这边是完成客户端那方的内容&#xff0c;当客户端登录成功之后&#xff0c;我们就要从服务器获…

redis实现消息队列的几种方式

一、了解 众所周知&#xff0c;redis是我们日常开发过程中使用最多的非关系型数据库&#xff0c;也是消息中间件。实际上除了常用的rabbitmq、rocketmq、kafka消息队列&#xff08;大家自己下去研究吧~模式都是通用的&#xff09;&#xff0c;我们也能使用redis实现消息队列。…

单片机智能家居火灾环境安全检测

目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 电路图采用Altium Designer进行设计&#xff1a; 三、实物设计图 四、程序源代码设计 五、获取资料内容 前言 在现代社会&#xff0c;火灾安全始终是人们关注的重点问题。随着科技的不…

【目标检测】用YOLOv8-Segment训练语义分割数据集(保姆级教学)

前言 这篇教程会手把手带你用 YOLOv8-Segment 搭建一个属于自己的分割任务项目。从环境配置到数据集准备&#xff0c;再到模型训练和测试&#xff0c;所有步骤都有详细说明&#xff0c;适合初学者使用。你将学会如何安装必要的软件&#xff0c;标注自己的数据&#xff0c;并使…

mac2019环境 Airflow+hive+spark+hadoop本地环境安装

1 环境介绍 本地安装可分为两个部分&#xff0c;mac软件环境&#xff0c; python开发环境 ps: 安装过程参考chatgpt、csdn文章 1.1 mac软件环境 目标安装的的软件是hive、apache-spark、hadoop&#xff0c;但是这三个软件又依赖java(spark依赖&#xff09;、ssh&#xff08…

1.7 JS性能优化

从输入url到页面加载完成都做了些什么 输入 URL - 资源定位符 http://www.zhaowa.com - http 协议 域名解析 https://www.zhaowa.com > ip 1. 切HOST&#xff1f; > 浏览器缓存映射、系统、路由、运营商、根服务器 2. 实际的静态文件存放&#xff1f; 大流量 > 多个…

【Ansible常用命令+模块+Playbook+Roles】

Ansible 一、命令1.1 常用命令 二、模块2.1 shell模块2.2 复制模块2.3 用户模块2.4 软件包管理2.5 服务模块2.6 文件模块2.7 收集模块2.8 fetch2.9 cron2.10 group2.11 script2.12 unarchive 三、YAML Roles3.1 目录结构3.2 文件内容tasks/main.yamlnginx.conf.j2vars/main.yam…

Oracle19C AWR报告分析之Wait Classes by Total Wait Time

Oracle19C AWR报告分析之Wait Classes by Total Wait Time 一、分析数据二、详细分析2.1 指标参数介绍2.2 数据库性能分析2.3 综合性能评估 在 Oracle 数据库的 AWR 报告中&#xff0c;Wait Classes by Total Wait Time 是评估数据库性能的重要部分。本篇文章主要是介绍指标参数…

嵌入式硬件电子电路设计(五)MOS管详解(NMOS、PMOS、三极管跟mos管的区别)

引言&#xff1a;在我们的日常使用中&#xff0c;MOS就是个纯粹的电子开关&#xff0c;虽然MOS管也有放大作用&#xff0c;但是几乎用不到&#xff0c;只用它的开关作用&#xff0c;一般的电机驱动&#xff0c;开关电源&#xff0c;逆变器等大功率设备&#xff0c;全部使用MOS管…

问题大集-01-kafka问题

1、问题&#xff1a;Windows下启动单机kafka出现&#xff1a;系统找不到指定路径 解决&#xff1a; 是kafka不能识别本机的java环境&#xff08;JVM&#xff09;&#xff0c;故需要指定java路径&#xff0c; 进入kafka路径下的\bin\windows&#xff0c;找到&#xff1a;kafk…

C++ 的发展

目录 C 的发展总结&#xff1a;​编辑 1. C 的早期发展&#xff08;1979-1985&#xff09; 2. C 标准化过程&#xff08;1985-1998&#xff09; 3. C 标准演化&#xff08;2003-2011&#xff09; 4. C11&#xff08;2011年&#xff09; 5. C14&#xff08;2014年&#xf…

Ubuntu问题 -- 允许ssh使用root用户登陆

目的 新重装的系统, 普通用户可以使用ssh登陆服务器, 但是root不能使用ssh登陆 方法 vim 编辑ssh配置文件 sudo vim /etc/ssh/sshd_config找到 PermitRootLogin 这一行, 把后面值改成 yes 重启ssh sudo service sshd restart然后使用root账号登陆即可