Ⅰ . list 的介绍和使用
01 初识 list
我们已经学习过 string 和 vector 了,想必大家已经掌握了查文档的能力
现在我们去学习如何使用 list ,最好仍然打开文档去学习
data:image/s3,"s3://crabby-images/33e98/33e985ac44c4d9f9bac0a0c9cc6eadd905b29dda" alt=""
data:image/s3,"s3://crabby-images/1d39e/1d39e1cb7a6f880b953f666c700ded55b5600d3e" alt=""
① list 是一个顺序容器
允许在任意位置进行 O(1) 插入和删除的顺序容器,并提供双向迭代器
② list 底层是双向链表
双向链表中每个元素存储在互不相关的独立结点中,在结点中通过两个指针指向其前后元素
③ list 与 forward_list 非常相似
最大的不同就是 forward_list 是单链表,只能向前迭代
④ 与其他序列式容器相比(array,vector,deque)
list 通常在任意位置进行插入、删除元素的效率更高,因为是 O(1)
list 和 forward_list 最大的缺陷是不支持随机位置的随机访问 举个例子:
如果要访问 list 中的第六个元素,必须从已知位置(如头部或尾部)迭代到该位置
在这段位置上迭代需要线性时间的时间开销,同时,list 还需要一些额外空间
以保存每个结点的相关联信息
02 创建 list
代码实现:
cpp
#include<iostream>
#include<list>
using namespace std;
int main()
{
list<int> l;
return 0;
}
Ⅱ . list 的修改操作
01 修改操作的函数
data:image/s3,"s3://crabby-images/a2ea9/a2ea917fb8c7a5d0963d6780a4ea5dedf5b8b065" alt=""
02 push_back
在 list 尾部插入值为 val 的元素
代码实现:
cpp
#include<iostream>
#include<list>
using namespace std;
int main()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
return 0;
}
我们现在来打印一下
首先思考一个问题,还能使用下标 + 方括号的方式遍历嘛?
不行 因为 list 是链表,是通过指针连接的,所以不支持随机访问
而 string 和 vector 可以,因为它们的物理结构是连续的
迭代器遍历:
cpp
#include<iostream>
#include<list>
using namespace std;
int main()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
// 迭代器
list<int>::iterator it = l.begin();
while (it != l.end())
{
cout << *it << " ";
++it;
}
cout << endl;
return 0;
}
运行结果如下:
data:image/s3,"s3://crabby-images/d58bb/d58bbd340f7104a41480a31e2321561fd6cc8b35" alt=""
范围 for 就更简单了:
cpp
// 范围for
for (auto e : l)
cout << e << " ";
cout << endl;
如果我们想倒着遍历,我们可以使用反向迭代器
使用 rbegin() 和 rend()
反向迭代器:
cpp
// 反向迭代器
list<int>::reverse_iterator rit = l.rbegin();
while (rit != l.rend())
{
cout << *rit << " ";
++rit;
}
cout << endl;
运行结果如下:
data:image/s3,"s3://crabby-images/dd004/dd0040741b63315365807d4431716d6e9bec45cc" alt=""
03 push_front
在 list 头部插入值为 val 的元素
代码实现:
cpp
void list_test2()
{
list<int> l;
l.push_front(1);
l.push_front(2);
l.push_front(3);
l.push_front(4);
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/f0943/f09438b9ff64aee5f814518dc99c6b56b01ee49c" alt=""
04 pop_back
删除 list 中的最后一个元素
cpp
void list_test3()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
cout << "删除前:";
for (auto e : l)
cout << e << " ";
cout << endl;
l.pop_back();
cout << "删除后:";
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/921b9/921b94f87fd4f82a440d021381151b301e848dcb" alt=""
如果表内没有元素,进行删除操作,会触发断言
05 pop_front
删除 list 中的第一个元素
cpp
void list_test4()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
cout << "删除前:";
for (auto e : l)
cout << e << " ";
cout << endl;
l.pop_front();
cout << "删除后:";
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/6746f/6746f8a63c48e9f2285bba38f76b538fe71d02cc" alt=""
06 insert
data:image/s3,"s3://crabby-images/6f7a6/6f7a62b1c9a9ebd7b047424627a1792f1265a6e3" alt=""
在上一节讲解 vector 实现的时候,我们对迭代器失效的问题进行了简单探讨
这里 list 的 insert 同样会涉及迭代器失效的问题,我们在模拟实现的时候再进行探讨
07 clear
data:image/s3,"s3://crabby-images/09ffb/09ffb544878e346ca75ba9981a0edb35afca131a" alt=""
清空 list 中的有效元素,使容器大小的 size 变为 0
代码演示:
cpp
void list_test5()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
cout << "清空前:";
for (auto e : l)
cout << e << " ";
cout << endl;
l.clear();
cout << "清空后:";
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/787e4/787e4f7ca2a2945129636978cf7497c4a6c0a5d0" alt=""
08 erase
data:image/s3,"s3://crabby-images/2b89e/2b89e65333988aea79698ecc0ad64e02ba2db8aa" alt=""
代码演示:
cpp
void list_test6()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
for (auto e : l)
cout << e << " ";
cout << endl;
list<int>::iterator it = find(l.begin(), l.end(), 4);
if (it != l.end())
{
l.erase(it);
}
else
{
cout << "没找到";
}
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/d219a/d219acb124ca715d777e2a08d31ca5e1182110e5" alt=""
Ⅲ . list 容量操作
01 size 返回有效结点个数
代码演示:
cpp
void list_test7()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
for (auto e : l)
cout << e << " ";
cout << endl;
cout << "有效节点个数:";
cout << l.size() << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/20a01/20a0127e8485a62100e24f090ee5a59cdb820cff" alt=""
02 empty 检测容器是否为空
data:image/s3,"s3://crabby-images/1ddef/1ddef4ee8e2d9d9e4003acb56b3a44dce8ddef1c" alt=""
empty 是用来检测容器是否为空的,如果为空则返回 true,否则返回 false。
代码演示:
cpp
void list_test8()
{
list<int> l;
l.empty() == true ? cout << "为空" : cout << "不为空";
}
运行结果如下:
data:image/s3,"s3://crabby-images/44134/44134e52f01c534bec3284929e5984afb6d38791" alt=""
03 resize 调整容器大小
data:image/s3,"s3://crabby-images/f1979/f197947149fbc73b7677aba2d95a6c54223cd658" alt=""
list 中的 resize 和之前的 resize 的 "扩容" 有点不一样,它没有容量。
这里的 resize 是调整容器的大小,使其包含 个元素。
代码演示:
cpp
void list_test9()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
for (auto e : l)
cout << e << " ";
cout << endl;
l.resize(10, 5);
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/35baf/35baffb481f6e1770950ec5fc79c2d4942a2ba33" alt=""
Ⅳ . list 的其他操作
01 reverse 逆置
data:image/s3,"s3://crabby-images/dc12b/dc12bf22fde73a8dac7cb94c8e189b6e895234df" alt=""
代码演示:
cpp
void list_test10()
{
list<int> l;
l.push_back(1);
l.push_back(2);
l.push_back(3);
l.push_back(4);
for (auto e : l)
cout << e << " ";
cout << endl;
l.reverse();
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/cd74f/cd74ffd2c757e3e4065329777553b8db227555da" alt=""
02 sort 排序
data:image/s3,"s3://crabby-images/65286/65286c1ee35f543bd6cfe95233597be0bf940460" alt=""
代码演示:
cpp
void list_test11()
{
list<int> l;
l.push_back(4);
l.push_back(2);
l.push_back(6);
l.push_back(1);
for (auto e : l)
cout << e << " ";
cout << endl;
l.sort();
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/35b11/35b1123dfdb3542cf4f51aadbf2a7e7dffc64d16" alt=""
03 unique 去重
data:image/s3,"s3://crabby-images/0fd03/0fd0316099cdaa5234fb82b3f7468fa4cb3b7055" alt=""
去重之前是有要求的,去重之前一定要先排序,如果不排序可能会去不干净
代码演示:
cpp
void list_test12()
{
list<int> l;
l.push_back(2);
l.push_back(1);
l.push_back(2);
l.push_back(1);
for (auto e : l)
cout << e << " ";
cout << endl;
l.sort();
cout << "去重后:";
l.unique();
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/9b836/9b8363704fe5d04975a1bbe23908995615ba7906" alt=""
如果不排序,会导致去不干净:
cpp
void list_test12()
{
list<int> l;
l.push_back(2);
l.push_back(1);
l.push_back(2);
l.push_back(1);
for (auto e : l)
cout << e << " ";
cout << endl;
//l.sort();
cout << "去重后:";
l.unique();
for (auto e : l)
cout << e << " ";
cout << endl;
}
data:image/s3,"s3://crabby-images/b98a4/b98a4fd9e779236d4223af5ff7da6f2a606b57f5" alt=""
04 remove
data:image/s3,"s3://crabby-images/777f4/777f42dda3a8bbc64ae101fbf7e51afb8b99f279" alt=""
remove 只需要给一个元素的值,它就可以自己找自己删
代码演示:
cpp
void list_test13()
{
list<int> l;
l.push_back(10);
l.push_back(20);
l.push_back(30);
l.push_back(40);
for (auto e : l)
cout << e << " ";
cout << endl;
// 如果删一个存在的元素
l.remove(10);
for (auto e : l)
cout << e << " ";
cout << endl;
// 如果待删元素不存在
l.remove(0);
for (auto e : l)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/0de51/0de510ba1a9cd35c561ac8a676ecbbb59d18f766" alt=""
05 splice 接合
data:image/s3,"s3://crabby-images/b5405/b54051a5d93e21dcb6e3743e57a7b4239bf7f722" alt=""
简单来说就是把一个链表转移到另一个链表中
代码演示:
cpp
void list_test14()
{
list<int> l1;
l1.push_back(1);
l1.push_back(2);
l1.push_back(3);
cout << "l1:";
for (auto e : l1)
cout << e << " ";
cout << endl;
list<int> l2;
l2.push_back(10);
l2.push_back(20);
l2.push_back(30);
cout << "l2:";
for (auto e : l2)
cout << e << " ";
cout << endl;
list<int>::iterator pos = l1.begin();
// 把l2的内容接到l1的begin()前面
l1.splice(pos, l2);
cout << "接合后";
for (auto e : l1)
cout << e << " ";
cout << endl;
}
运行结果如下:
data:image/s3,"s3://crabby-images/38a30/38a30e9c9e1a83d9743d7688049edc0c4c5855d3" alt=""