List基本使用(C++)

news2024/9/8 10:28:48

目录

1.list的介绍

2.list的使用

list的构造

list的size() 和 max_size()

list遍历操作

list元素修改操作

assign()函数

push_front(),push_back 头插,尾插

pop_front() pop_back 头删尾删

insert()函数

swap()函数

resize()函数

clear()函数

list类数据操作

splice()函数

sort()函数

unique()函数

reverse()函数(逆置链表)


1.list的介绍

1.list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。

2.list的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。

3.list与forward_list非常相似:最主要的不同在于forward_list是单链表,只能朝前迭代,已让其更简单高效。

4.与其他的序列式容器相比(array,vector,deque),list通常在任意位置进行插入、移动元素的执行效率更好。

5.与其他序列式容器相比,list和forward_list最大的缺陷是不支持任意位置的随机访问,比如:要访问list的第六个元素,必须从已知的位置(比如头部或者尾部)迭代到该位置,在这段位置上迭代需要线性的时间开销;list还需要一些额外的空间,以保存每个节点的相关联信息(对于存储类型较小元素的大list来说这可能是一个重要的因素)

2.list的使用

list类为类模板,所以在使用时需要带上类型表示一个具体的类,例如数据类型为int类型的list使用时需要写为list<int>

list的构造

#include <iostream>
using namespace std;
#include <list>
#include <vector>
int main() {
    //1.无参构造
    list<int> ls1;
        //list没有capacity
    cout << ls1.size() << endl;//0
    
    //2.指定个数类型值进行构造
    list<int> ls2(4,5);
    cout << ls2.size() <<endl;//4
    for(auto e : ls2){
        cout << e << " ";
    }
    cout << endl;//5 5 5 5
    
    //3.使用类对象迭代器区间进行构造
    
    list<int> ls3(ls2.begin(),ls2.end());
    cout << ls3.size() <<endl;//4
    for(auto e : ls3){
        cout << e << " ";
    }
    cout << endl;//5 5 5 5
    
    //也可以使用其然类对象迭代器区间进行构造
    vector<int> v(4,5);
    list<int> ls4(v.begin(),v.end());
    cout << ls4.size() <<endl;//4
    for(auto e : ls4){
        cout << e << " ";
    }
    cout << endl;//5 5 5 5

    //4.拷贝构造
    list<int> ls5(ls1);
    //list<int> ls5 = ls1;
    cout << ls5.size() << endl;//0
    return 0;
}

list的size() 和 max_size()

list遍历操作

在list中,只有迭代器遍历方式

list元素修改操作

assign()函数

使用assign()函数可以为调用对象链表重新分配内容,如果原始链表中有数据,那么将覆盖原始内容

int main(){
    list<int> ls1(4,8);
    list<int> ls2 = {1,2,3,4,5};
    
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//8 8 8 8
    
    ls1.assign(ls2.begin(),ls2.end());
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//1 2 3 4 5
    
    return 0;
}

push_front(),push_back 头插,尾插

int main(){
    list<int> ls1 = {1,2,3,4,5};
    
    ls1.push_front(0);
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//0 1 2 3 4 5
    
    ls1.push_back(6);
    ls1.push_back(7);
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;//0 1 2 3 4 5 6 7
    
    return 0;
}

pop_front() pop_back 头删尾删

int main(){
    list<int> ls1 = {1,2,3,4,5};
    
    ls1.pop_front();
    ls1.pop_front();
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//3 4 5
    
    ls1.pop_back();
    ls1.pop_back();
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//3 
    return 0;
}

insert()函数

对于insert()函数来说,基本不存在迭代器失效问题,因为list不存在扩容问题并且空间基本不是连续的,所以pos位置在插入数据后可能并没有改变

int main(){
    list<int> ls1 = {1,2,3,4,5};
    
    ls1.insert(ls1.end(), 6);
    ls1.insert(ls1.begin(), 2, 0);
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//0 0 1 2 3 4 5 6
    
    list<int> ls2 = {1,3,1,4};
    ls1.insert(ls1.begin(), ls2.begin(), ls2.end());
    for(auto e : ls1)
        cout << e << " ";
    cout << endl;//1 3 1 4 0 0 1 2 3 4 5 6 
    
    return 0;
}

int main(){
    list<int> ls1 = {1,2,3,4,5};
    list<int>::iterator it =  find(ls1.begin(), ls1.end(), 2);
        it = ls1.erase(it);
        for (auto e : ls1)
        {
            cout << e << " ";
        }
    //1 3 4 5
     
        cout << endl << *it << endl;
    //3
    return 0;
}

swap()函数

交换两个链表

int main(){
    list<int> ls1 = {1,2,3,4,5};
    cout << "ls1: ";
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    
    list<int> ls2 = {5,4,3,2,1};
    cout << "ls2: ";
    for(auto e : ls2)
    cout << e << " ";
    cout << endl;
    
    swap(ls1, ls2);
    cout << "ls1: ";
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    
    cout << "ls2: ";
    for(auto e : ls2)
    cout << e << " ";
    cout << endl;
    return 0;
}

resize()函数

修改指定对象链表中的有效数据节点的个数

当n小于当前链表的size,则实现删除效果,否则初始化为指定类型的数据(默认为对应类型的默认值)

int main(){
    list<int> ls1 = {1,2,3,4,5};
    cout << "ls1 size:" << ls1.size() << endl;
    
    //设定大小大于原来size,默认补充
    ls1.resize(10);
    cout << "ls1 size:" << ls1.size() << endl;
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    
    
    //设定大小小于原来size,相当于删除
    ls1.resize(2);
    cout << "ls1 size:" << ls1.size() << endl;
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    return 0;
}
输出结果:
ls1 size:5
ls1 size:10
1 2 3 4 5 0 0 0 0 0 
ls1 size:2
1 2

clear()函数

使用clear()函数可以清空调用对象链表当前所有的有效数据节点(不会删除头节点)

int main(){
    list<int> ls1 = {1,2,3,4,5};
    cout << "清空前:" << endl;
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    
    ls1.clear();
    cout << "清空后:" << endl;
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;
    return 0;
}

清空前:
1 2 3 4 5 
清空后:

list类数据操作

splice()函数

使用splice()函数可以将指定对象中的内容拼接到调用对象的链表中的指定位置后(拼结后原链表不存在,只剩拼接后的新链表)

int main(){
    list<int> ls1 = {1,3,1,4};
    list<int> ls2 = {5,2,0};
    
    ls1.splice(ls1.begin(), ls2);
    for(auto e : ls1)
    cout << e << " ";
    cout << endl;//5 2 0 1 3 1 4
    
    for(auto e : ls2)
    cout << e << " ";
    cout << endl;//ls2 内容为空
    
    return 0;
}

注意,splice()函数中的迭代器为双向迭代器(bidirectional iterator),传递的迭代器也必须为双向迭代器或者单向迭代器(forward iterator),不可以随机迭代器(random iterator)

随机访问迭代器是特殊的双向迭代器,双向迭代器和随机访问迭代器是特殊的单向迭代器

随机访问迭代器支持以下操作:

双向迭代器支持以下操作:

单向迭代器支持以下操作:

#include <iostream>
#include <list>
#include <vector>
using namespace std;
 
int main()
{
    list<int> ls{ 1,2,3,4,5 };
    vector<int> v1(3, 6);
 
    cout << "拼接前:" << endl;
    for (auto num : ls)
    {
        cout << num << " ";
    }
 
    cout << endl;
 
    for (auto num : v1)
    {
        cout << num << " ";
    }
 
    cout << endl;
    list<int>::iterator it = find(ls.begin(), ls.end(), 2);
    ls.splice(it, v1);
    cout << "拼接后:" << endl;
    for (auto num : ls)
    {
        cout << num << " ";
    }
 
    cout << endl;
 
    for (auto num : v1)
    {
        cout << num << " ";
    }
 
    cout << endl;
    return 0;
}
报错信息:
“std::list<int,std::allocator<int>>::splice”: 没有重载函数可以转换所有参数类型

因为vector的迭代器为随机访问迭代器,所以当传入双向迭代器时会报错

sort()函数

使用sort()函数可以为调用对象的链表进行排序,底层是归并排序

默认是升序排序,可以通过仿函数改变为降序(后续介绍仿函数)

int main()
{
    //升序
    list<int> ls = {9,0,3,2,21,43,11,0};
    for (auto num : ls)
    {
        cout << num << " ";
    }
    cout << endl;
    ls.sort();
    for (auto num : ls)
    {
        cout << num << " ";
    }
    cout << endl;
    //降序
    ls.sort(greater<int>());
    for (auto num : ls)
    {
        cout << num << " ";
    }
    return 0;
}
9 0 3 2 21 43 11 0 
0 0 2 3 9 11 21 43 
43 21 11 9 3 2 0 0

当list排序数据非常多的时候,可以考略将list数据拷贝到vector中排序,排序后再拷回list中。这段过程看似麻烦,但总体消耗时间比单独使用list中sort()函数小

int main()
{
    srand(time(NULL));
    const int N = 10000;//一万个数据
 
    list<int> ls;
    list<int> ls1;
 
    //向两个链表中插入数据
    for (int i = 0; i < N; ++i)
    {
        auto e = rand();
        ls.push_back(e);
        ls1.push_back(e);
    }
 
    //直接使用sort()排序
    size_t begin = clock();
    ls.sort();
    size_t end = clock();
 
    //拷贝到vector类中排序
    size_t begin1 = clock();
    vector<int> v(ls1.begin(), ls1.end());
    sort(v.begin(), v.end());
    ls1.assign(v.begin(), v.end());
    size_t end1 = clock();
 
    cout << "直接排序:" << end - begin << "ms" << endl;
    cout << "拷贝后排序再拷贝:" << end1 - begin1 << "ms" << endl;
 
    return 0;
}
输出结果:
直接排序:2934ms
拷贝后排序再拷贝:586ms

unique()函数

unique()函数可以为链表去除重复数据的有效数据节点,但是使用unique()函数之前必须确保链表有序

int main(){
    list<int> ls = {9,0,3,2,2,21,43,11,0};
    //进行升序
    ls.sort();
    for (auto e : ls)
    cout << e << " ";
    cout << endl;
    
    ls.unique();
    for (auto e : ls)
    cout << e << " ";
    cout << endl;
    return 0;
}
输出结果;
0 0 2 2 3 9 11 21 43 
0 2 3 9 11 21 43 

reverse()函数(逆置链表)

int main(){
    list<int> ls = {1,2,3,4,5,6};
    ls.reverse();
    for (auto e : ls)
    cout << e << " ";
    cout << endl;//6 5 4 3 2 1 
    return 0;
}

 

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

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

相关文章

yolov10 瑞芯微 rknn 部署 C++代码

yolov10 目标检测rknn的C部署来了。 特别说明&#xff1a;如有侵权告知删除&#xff0c;谢谢。 直接上代码和模型&#xff0c;欢迎参考交流 【完整代码和模型】 1、rknn模型准备 pytorch转onnx&#xff0c;onnx再转rknn模型这一步就不再赘述&#xff0c;请参考上一篇 【yolov1…

TPM之VMK密封

本篇文章主要介绍基于TPM的Bitlocker全盘加密时,VMK密钥的密封(Seal)流程,至于TPM、Bitlocker、密钥保护器、VMK密钥等这些东西是什么,这里不做解释,需要自己脑补一下(╮(╯▽╰)╭)。 首先看看一张结构图(来自网络),了解一下TPM加密的基本框架与流程: 同样,基于…

关于读书,你可能没想到的陷阱、问题和思考

最近经常看到有人问&#xff1a;AI 已经这么发达了&#xff0c;如果以后的 AI 会更智能&#xff0c;那我们还有必要读书吗&#xff1f; 我认为&#xff1a;还是十分有必要的。 为什么呢&#xff1f;因为读书其实不仅仅是为了获取知识&#xff0c;它更重要的一个作用&#xff0c…

数据结构-堆排序问题

需要在数组里面进行排序&#xff0c;我们可以采取堆排序对其解决问题 版本1&#xff1a; 创建一个数组等大的堆&#xff0c;把数组里面的数值输入到堆里面进行堆排序&#xff0c;但是这样的弊端就是&#xff0c;不是顺序排序 版本2&#xff1a; 每次我们取堆顶然后打印&#xf…

3440亿!国家大基金三期正式落地,关注半导体与算力芯片!

重磅消息来了&#xff01; 5月24日&#xff0c;注册规模3,440亿元人民币的“国家集成电路产业投资基金三期股份有限公司”正式成立&#xff0c;这也意味着&#xff0c;传闻已久的**“国家大基金三期”正式落地&#xff01;** 企查查股东信息显示&#xff0c;该公司由财政部、国…

移动云:开发者手中的未来钥匙

《移动云&#xff1a;开发者手中的未来钥匙》 引言一、无缝集成&#xff0c;加速开发进程二、数据智能&#xff0c;洞悉用户心声三、安全合规&#xff0c;护航创新之旅四、成本优化&#xff0c;助力轻装前行总结 引言 在科技日新月异的今天&#xff0c;移动云已成为推动行业变革…

文件夹类型异常成文件:原因解析与恢复策略

在数字时代&#xff0c;数据的安全与完整性对于个人和企业都至关重要。然而&#xff0c;有时我们可能会遇到一种令人困惑的情况&#xff1a;原本应该是文件夹的图标&#xff0c;却突然变成了文件的图标&#xff0c;这就是所谓的“文件夹类型成文件”问题。本文将深入探讨这一现…

(原创)从右到左排列RecycleView的数据

问题的提出 当我们写一个Recycleview时&#xff0c;默认的效果大概是这样的&#xff1a; 当然&#xff0c;我们也可以用表格布局管理器GridLayoutManager做成这样&#xff1a; 可以看到&#xff0c;默认的绘制方向是&#xff1a; 从左到右&#xff0c;从上到下 那么问题来了…

香橙派 AIpro综合体验及AI样例运行

香橙派 AIpro综合体验及AI样例运行 环境&#xff1a; 香橙派版本&#xff1a; AIpro(8TOPSINT8) OS : Ubuntu 22.04.3 LTS(GNU/Linux 5.10.0 aarch64) (2024-03-18) 远程服务端1&#xff1a;OpenSSH 8.9p1 远程服务端2&#xff1a;TightVNC Server 1.3.10 远程客户端&#xf…

使用numpy手写一个神经网络

本文主要包含以下内容&#xff1a; 推导神经网络的误差反向传播过程使用numpy编写简单的神经网络&#xff0c;并使用iris数据集和california_housing数据集分别进行分类和回归任务&#xff0c;最终将训练过程可视化。 1. BP算法的推导过程 1.1 导入 前向传播和反向传播的总体…

基于EBAZ4205矿板的图像处理:10gamma变换

基于EBAZ4205矿板的图像处理&#xff1a;10gamma变换 项目全部文件 会上传项目全部文件&#xff0c;如果没传&#xff0c;可以私信催我一下&#xff0c;最近太忙了 先看效果 我的项目中的gamma的变换系数为2.2&#xff0c;是会让图像整体变暗的&#xff0c;看右图说明我的ga…

哪款洗地机好用?洗地机十大排行榜

在智能家电飞速发展的今天&#xff0c;洗地机因其吸拖洗一体化的技术优势&#xff0c;成为越来越多家庭的清洁利器。它不仅能快速清理各种地面污渍&#xff0c;还能轻松处理干湿垃圾&#xff0c;大大提升了日常清洁的效率。可是面对市场上琳琅满目的洗地机品牌和型号&#xff0…

数据持久化第六课-ASP.NET运行机制

数据持久化第六课-ASP.NET运行机制 一.预习笔记 1.动态网页的工作机制通常分为以下几个阶段&#xff1a; 1&#xff09;使用动态Web开发技术编写Web应用程序&#xff0c;并部署到Web服务器。 2&#xff09;客户端通过在浏览器中输入地址&#xff0c;请求动态页面。 3&#…

Swift 初学者交心:在 Array 和 Set 之间我们该如何抉择?

概述 初学 Swift 且头发茂密的小码农们在日常开发中必定会在数组&#xff08;Array&#xff09;和集合&#xff08;Set&#xff09;两种类型之间的选择中“摇摆不定”&#xff0c;这也是人之常情。 Array 和 Set 在某些方面“亲如兄弟”&#xff0c;但实际上它们之间却有着“云…

关于DDos防御...别在听别人瞎扯了.....

前言 无意间刷文章的时候看到一篇文章&#xff0c;写的是遇到ddos&#xff0c;怎么用iptables封IP....... 然后我就百度搜了一下&#xff0c;好多都是这么说的&#xff0c;但是我发现&#xff0c;大多数人只要遭受过长期Ddos的&#xff0c;就不会再信网上的文章 文笔不太好&…

【Qt】深入探索Qt事件处理:从基础到高级自定义:QEvent

文章目录 前言&#xff1a;1. 事件的介绍2. 事件的处理2.1. 示例1&#xff1a; 重写鼠标进入和鼠标离开事件2.2. 示例2&#xff1a;当鼠标点击时&#xff0c;获取对应的坐标值&#xff1b;2.3. 鼠标释放事件2.4. 鼠标双击事件2.5. 鼠标移动事件2.6. 鼠标滚轮的滚动事件 3. 按键…

后端经典三层架构

大家好&#xff0c;这里是教授.F 引入&#xff1a; MVC 全称∶ Model 模型、View 视图、 Controller 控制器。MVC 最早出现在 JavaEE 三层中的 Web 层&#xff0c;它可以有效的指导WEB 层的代码如何有效分离&#xff0c;单独工作。 View 视图∶只负责数据和界面的显示&#…

Python I/O操作笔记

打开文件&#xff1a; 使用 open() 函数&#xff0c;其中文件路径可以是相对路径或绝对路径。 模式除了常见的 r&#xff08;只读&#xff09;、w&#xff08;写入&#xff0c;会覆盖原有内容&#xff09;、a&#xff08;追加&#xff09;外&#xff0c;还有一些其他组合模式&…

小度推出全球首款基于文心大模型的学习机Z30,仅售价6699元

5月27日&#xff0c;小度科技召开新品发布会&#xff0c;全球首款基于文心大模型的学习机——小度学习机Z30重磅发布。 据「TMT星球」了解&#xff0c;该产品基于文心大模型&#xff0c;重新定义了“AI老师”的能力边界&#xff0c;不仅是一款能为孩子提供全面、有效学习辅导的…

LINUX环境基础练习题(附带答案)

&#x1f525; 交流讨论&#xff1a;欢迎加入我们一起学习&#xff01; &#x1f525; 资源分享&#xff1a;耗时200小时精选的「软件测试」资料包 &#x1f525; 教程推荐&#xff1a;火遍全网的《软件测试》教程 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1…