C++STL详解(五)——list的介绍与使用

news2025/1/25 9:20:57

文章目录

  • list的介绍
  • list的使用
      • list的定义方法
      • list迭代器失效问题
      • list插入和删除
          • insert
          • erase
      • list迭代器的使用
          • begin,end 和 rbegin,rend
      • list元素访问
          • front 和 back
      • list容量控制与数据清理
          • resize
          • clear
      • list操作函数
          • splice
          • remove 和 remove_if
          • unique
          • merge
          • reverse
          • assign

list的介绍

1:list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。
2:与其他序列容器相比(array,vector,deque),
list通常可以在任意位置进行插入,移除等效率更高。
3:与其他序列式容器相比,list和forward_list最大的缺陷就是不支持任意位置的访问,比如:要访问list的第6个元素,必须从已知的位置的位置迭代到目标位置,然而,迭代的时间复杂度通常为线性开销,list还需要一些额外的空间,来保存每个结点的相关联信息(对于存储类型较小的list来说)。

list的使用

list的定义方法

方法一

list<int> lt;

方法二
构造1个2类型为list 容器。

list<int> lt1(1,2);

方法三
拷贝构造类型为list 的lt1。

list<int> lt2( lt1);

方法四
利用迭代器区间进行构造。

string s("hello c++");
list<char>lt3(s.begin(),s.end());

方法五
也是迭代器区间构造的一种形式,区间为左闭右开。


	int arr[] = { 1,2,3,4,5 };
	int sz = sizeof(arr) / sizeof(int);

	list<int> lt1(arr, arr + 2);

list迭代器失效问题

void Test1()
{
 int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };
 list<int> l(array, array+sizeof(array)/sizeof(array[0]));
 auto it = l.begin();
 while (it != l.end())
 {
 // erase()函数执行后,it所指向的节点已被删除,因此it无效,在下一次使用it时,必须先给
其赋值
 l.erase(it); 
 ++it;
 }
}

list插入和删除

insert

三种插入方式
方式1:
在指定位置前插入一个数。
方式2:
在指定位置插入n个数。
方式3:
在指定位置插入相同容器而理性的迭代器区间数据(左闭右开)

int main()
{

	list<int> lt;
	lt.push_back(1);
	lt.push_back(2);
	lt.push_back(3);
	list<int>::iterator pos = find(lt.begin(), lt.end(),3 );
	lt.insert(pos, 0); 
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 
	pos = find(lt.begin(), lt.end(), 3);
	lt.insert(pos, 2, 1); //
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 
	vector<int> v(3, 8);
	pos = find(lt.begin(), lt.end(), 1);
	lt.insert(pos, v.begin(), v.end());
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 

}
erase

三种删除方式
方式1:
指定目标位置进行删除。
方式2:
指定迭代器区间进行删除。

int main()
{
	list<int> lt;
	lt.push_back(1);
	lt.push_back(2);
	lt.push_back(3);
	lt.push_back(4);
	lt.push_back(5);
	lt.push_back(6);
	auto pos = find(lt.begin(), lt.end(), 2);
	lt.erase(pos); //删除2位置的结点。
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 
	pos = find(lt.begin(), lt.end(), 3);
	lt.erase(pos, lt.end()); //给定迭代器区间删除3后面的结点。
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 
	return 0;
}

list迭代器的使用

在这里插入图片描述
正向迭代器和反向迭代器的位置是对称的,指向的位置相反。

begin,end 和 rbegin,rend
int main()
{
	list<int> lt{ 1,2,3,4,5 };
	//正向迭代器遍历容器
	auto it = lt.begin();
	while (it != lt.end())
	{
		cout << *it << " ";
		it++;
	}
	cout << endl;

	auto  rit = lt.rbegin();
	while( rit != lt.rend() )
	{
		cout << *rit << " ";
		rit++;
	}
	
	return 0;

}

list元素访问

由于list和forward_lsit一样都不能随机访问,STL中具有获取list首尾元素功能。

front 和 back
int main()
{
	list<int> lt;
	lt.push_back(0);
	lt.push_back(1);
	lt.push_back(2);
	lt.push_back(3);
	lt.push_back(4);
	lt.push_back(5);
	lt.push_back(6);
	cout << lt.front() << endl;     //获取list首元素
	cout << lt.back() << endl;      //获取list尾元素
	return 0;
}

list容量控制与数据清理

resize

1: 如果所给的第一个值大于lsit当前的size,如果给了第二个值,那么多出的size的值就为第二个所给值,如果没给,编译器就主动调用缺省值(一般为0)。
2:如果所给的值小于list当前的size,则lsit当前的size就为这个所给值

int main()
{
	list<int> lt{ 1,2,23 };
	for (auto e : lt)
	{
		cout << e << " ";
	}
	lt.resize(7, 6); //多出的size初始值都为6;
	cout << endl;
	for (auto e : lt)
	{
		cout << e << " ";
	}
	lt.resize(1);      //list当前只剩下一个size;
	cout << endl;
	for (auto e : lt)
	{
		cout << e << " ";
	}
}
clear

clear用于清空list容器,此时list的size为0;

int main()
{
	list<int> lt(3, 2);
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl; 
	cout << lt.size() << endl; 
	lt.clear(); 
	for (auto e : lt)
	{
		cout << e << " ";
	}
	cout << endl;

	return 0;
}

list操作函数

splice

list中splice函数用于两个list容器之间的拼接,有三种常见拼接方法。
1: 将指定容器全部数据拼接到指定位置。
2: 将指定容器的某个数据拼接到目标容器的指定位置。
3: 将指定容器的某个迭代器区间(左闭右开)数据拼接到目标容器中。

int main()
{
	list<int> lt1(1, 1);
	list<int> lt2(1, 2);
	//将lt2拼接到lt1首部。
	lt1.splice(lt1.begin(), lt2); 
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
	cout << endl; 
	list<int> lt3(1, 3);
	list<int> lt4(1, 4);
	
	//将lt4容器中首位置拼接到目标容器lt3的尾部。
	lt3.splice(lt3.end(), lt4, lt4.begin()); 
    for(auto&e : lt3)
	{
		cout << e << " ";
	}
	cout << endl;

	list<int> lt5(1, 5);
	list<int> lt6(1, 6);
	//将指定容器lt6的指定迭代器区间内的数据拼接到目标容器lt5的首部
	lt5.splice(lt5.begin(), lt6, lt6.begin(), lt6.end()); 
	for (auto& e : lt5)
	{
		cout << e << " ";
	}
	cout << endl;
	return 0;
}
remove 和 remove_if

remove:
删除容器容器中指定的数据。(包括重复项)
remove_if
删除容器当中符合条件的数据。

bool test(const int& val)
{
	//删除中容器小于3的数。
    return val < 3;
	
}
int main()
{
	list<int> lt1{ 1,2,3,4,5 };
	lt1.remove_if(test);              
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
	list<int> lt2{ 1,2,3,4,5,6 };
	//删除list容器中指定元素。
	lt2.remove(3); 
	for (auto& e : lt2)
	{
		cout << e << " ";
	}

}
unique

删除容器中连续的重复元素。
注意
在使用unique去重之前最好先让list排序。

int main()
{
	list<int> lt1{ 3,1,1,2,3,3,4,5 };
	//在使用unique之前最好先使用sort排序一下。
	lt1.sort();
	lt1.unique();
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
}
merge

将指定容器合并到目标容器中,并且合并过后的容器lt1依然为升序(类似于归并排序)

int main()
{
	list<int> lt1{ 3,1,1,2,3,3,4,5 };
	//在使用unique之前最好先使用sort排序一下。
	list<int> lt2{ 6,7,8 };
	lt1.sort();
	//将lt2有序容器合并到lt1中。
	lt1.merge(lt2);
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
}
reverse

将容器中所有数据的位置进行逆置。

int main()
{
	list<int> lt1{ 1,2,3 };
	lt1.reverse();
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
}
assign

1:将指定个数指定内容的分配给容器(覆盖分配);
2:利用迭代器,将存储相同数据类型的容器,指定迭代器区间进行分配。

int main()
{
	list<int> lt1{ 1,2,3 };
	lt1.assign(3, 1);
	for (auto& e : lt1)
	{
		cout << e << " ";
	}
	string s("hello C++");
	list<char>lt2;
	lt2.assign(s.begin(), s.end());
	
	for (auto& e : lt2)
	{
		cout << e << " ";
	}
}

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

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

相关文章

安卓开发之动态设置网络访问地址

之前开发程序联测测接口的时候&#xff0c;因为要和不同的后台人员调接口&#xff0c;所以经常要先把程序里的ip地址改成后台人员给我的。每次都要先修改ip地址&#xff0c;之后编译运行一下&#xff0c;才能测试。但要是换了个后台人员&#xff0c;或者同时和2个后台人员测接口…

Android提词器实现富文本样式

前提前一段时间做了一个程序&#xff0c;提词器APP&#xff0c;结合greendao保存数据。最近新增了一个需求&#xff0c;实现部分文字富文本的展现。师傅找了一个网上的SDK&#xff0c;但是在集成的时候总是出问题&#xff0c;我又不想把项目挪进来&#xff0c;感觉很麻烦&#…

Oracle P6 Professional相比与Microsoft Project的8个优势

目录 引言 1. 自上而下的调度 2. 努力程度 (LOE) 活动 3. 最长路径 4. 多浮动路径分析功能 6.预算材料成本 7. 开始和完成里程碑 8. 工作公式类型 概括 引言 哪种日程安排工具更适合您的情况&#xff0c;Oracle Primavera P6 还是 Microsoft Project(MSP) 经常有一些…

MySQL8.0Linux安装及主从的搭建

MySQL8.0Linux安装教程 下载并安装 需要说明的一点是我使用的是SSH secure shell Client连接linux系统的&#xff0c;它的用法和命令窗口差不多。界面如图&#xff1a;一样的使用Linux命令操作。 话不多说 第一步&#xff1a; 1&#xff09;、切换到 /usr/local下 cd /usr/…

已解决hint : See above for output from the failure.

已解决&#xff08;pip install wxPython安装失败&#xff09;error: legacy-instal1-failure Encountered error while trying to install package.wxPython note: This is an issue with the package mentioned above&#xff0c;not pip. hint : See above for output from …

关于世界坐标系,相机坐标系,图像坐标系,像素坐标系的一些理解

关于世界坐标系&#xff0c;相机坐标系&#xff0c;图像坐标系&#xff0c;像素坐标系的一些理解前言一、各坐标系的含义二、坐标系转换1.世界坐标系与相机坐标系&#xff08;旋转与平移&#xff09;2.相机坐标系与图像坐标系&#xff08;透视&#xff09;3.图像坐标系与像素坐…

【UE4 RTS游戏】02-摄像机运动_完成摄像机在X轴上运动的相关步骤

效果通过控制键盘WS键使得“CameraPawn”进行前后移动步骤将landscape的Z轴位置更改为0删除“PostProcessVolume”将“LightmassImportanceVolume”移入Lighting文件夹内新建一个蓝图类&#xff0c;父类是Pawn&#xff0c;命名为“CameraPawn”将“MyController”重命名为“Cam…

详解JVM

详解JVM 最近学习了&#xff1a;周志明《深入理解高并发编程》&#xff1b;&#xff1b; 特此简要对学习做了部分总结&#xff0c;方便后续对JVM相关知识的完善和巩固&#xff1b; 若想深入了解学习&#xff0c;可阅读上述参考原著&#xff1b; Java内存区域与OOM 运行时数据…

大数据 | (三)centos7图形界面无法执行yum命令

大家好&#xff0c;今天是三八女神节了&#xff01; 你知道吗&#xff1f;世界上第一位电脑程序设计师是名女性&#xff0c;Ada Lovelace (1815-1852)。 她是一位英国数学家兼作家&#xff0c;第一位主张计算机不只可以用来算数的人&#xff0c;也发表了第一段分析机用的演算…

vector中迭代器失效的问题及解决办法

目录 vector常用接口 vector 迭代器失效问题 vector中深浅拷贝问题 vector的数据安排以及操作方式&#xff0c;与array非常相似。两者的唯一差别在于空间的运用的灵活性。array 是静态空间&#xff0c;一旦配置了就不能改变&#xff1b;要换个大(或小) 一点的房子&#x…

CorelDRAW Graphics Suite2023更新内容介绍

懂设计的职场人都知道这款软件&#xff0c;CorelDRAW是一款非常高效的矢量图形设计软件。CorelDRAW操作界面简洁易懂&#xff0c;能够为用户提供精确地创建物体的尺寸和位置的功能&#xff0c;减少点击步骤&#xff0c;提高设计效率&#xff0c;节省设计时间。功能比普通的美图…

简单理解TransFormer

背景:听了李宏毅老师关于transformer的讲解&#xff0c;觉得有必要记录一下&#xff0c;里面的PPT都是李宏毅老师的内容(不喜勿喷)1.self-attention在介绍transformer之前&#xff0c;必须先了解self-attention(1) 先将X输入Embedding(a Wx), 然后a乘相关的权重&#xff0c;生…

Day11-网页布局实战-CSS3动画

文章目录一 CSS3动画1 2D动画案例1-鼠标输入移入DIV 让图片旋转90度案例2-鼠标输入移入DIV 缩放图片案例3-贯穿项目-DIV移动2 animation动画播放器案例1-基础案例案例2-使用百分比关键帧定义动画案例3-旋转的图片案例4-贯穿案例-轮播图3 多余文本省略号...代替案例1-多余文本..…

一 Go环境搭建

1. 下载地址 https://golang.google.cn/dl/ 傻瓜式安装&#xff0c;自动会配置path的变量&#xff0c;安装完成后可以使用go version 查看当前安装的版本 本文使用目前最新的1.20.2版本 2. 配置go环境 cmd控制栏打开输入以下命令&#xff08;如果cmd有问题可以尝试powershe…

340秒语音芯片,轻松实现语音交互,畅享智能生活WTV380语音ic方案

随着智能家居、安防报警、宠物用品 等&#xff0c;智能设备的普及&#xff0c;语音交互技术正在逐渐成为人机交互的主要方式之一。而如何实现稳定高效的语音交互&#xff0c;就需要借助先进的语音芯片技术。今天&#xff0c;我们介绍的是一款高性能的语音芯片——WTV380&#x…

Gamma矫正

Gamma 曲线Gamma校正被使用在8位RGB图中。用来解决在有限的存储空间中保存尽可能多的人类感受敏感的色彩内容。Gamma 矫正Gamma校正的方式就是采样时,和输出到显示器给人类看时,对亮度进行的调整.如采样时 Gamma1/2.2 调亮Gamma&#xff0c;如显示时 Gamma2.2 调暗Gamma实际亮度…

【Redis】Redis慢查询

文章目录慢查询记录慢查询两个配置参数修改配置参数慢查询日志慢查询记录 我们都知道像mysql等持久化数据库会有慢查询日志&#xff0c;其实Redis中也有慢查询日志的功能。慢查询就是系统在执行命令的前后计算每条命令的执行时间&#xff0c;如果超过我们预设的时间&#xff0c…

登录接口-简约版(工作日记4)

前提条件&#xff1a; 1、jmeter接口测试工具 2、接口信息 获取验证码信息 登录接口信息 题外话&#xff1a; 接口返回的数据中如果有文字就会乱码 怎么解决 1、进入jmter文件–进入bin文件夹&#xff0c;找到jmeter.properties文件 2、右键打开&#xff0c;可以用文本模…

基于遗传算法的CVRP建模求解(Python)

带容量约束的车辆路径优化问题&#xff0c;CVRP&#xff0c;对一系列装卸货点进行适当的路径规划&#xff0c;在满足约束条件&#xff08;客户需求、车辆载重和容积、车型、车辆行驶里程、配送中心数量等限制&#xff09;和目标最优化&#xff08;路程最短、成本最低、使用车辆…

点灯大师--点个正点原子阿尔法开发板的灯

点灯大师–点个正点原子阿尔法开发板的灯 文章目录点灯大师--点个正点原子阿尔法开发板的灯正点原子阿尔法开发板点灯1、使能 GPIO1 时钟2、设置 GPIO1_IO03 的复用功能3、配置 GPIO1_IO034、设置 GPIO5、控制 GPIO 的输出电平五种点灯的方法1.在一个驱动文件中实现寄存器初始化…