【C++】list的使用

目录

    • [1 构造](#1 构造)
      • [1.1 无参构造](#1.1 无参构造)
      • [1.2 构造的list中包含n个值为val的元素](#1.2 构造的list中包含n个值为val的元素)
      • [1.3 用[first, last)区间中的元素构造list](#1.3 用[first, last)区间中的元素构造list)
      • [1.4 拷贝构造](#1.4 拷贝构造)
    • [2 迭代器的使用](#2 迭代器的使用)
      • [2.1 begin + end](#2.1 begin + end)
      • [2.2 rbegin + rend](#2.2 rbegin + rend)
    • [3 容量操作](#3 容量操作)
      • [3.1 empty + size](#3.1 empty + size)
    • [4 获取元素](#4 获取元素)
      • [4.1 front + back](#4.1 front + back)
    • [5 插入、删除、修改](#5 插入、删除、修改)
      • [5.1 头插-push_front和尾插-push_back](#5.1 头插-push_front和尾插-push_back)
      • [5.2 头删-pop_front和尾删-pop_back](#5.2 头删-pop_front和尾删-pop_back)
      • [5.3 交换------swap](#5.3 交换——swap)
      • [5.4 清理------clear](#5.4 清理——clear)
      • [5.5 insert-pos位置插入元素](#5.5 insert-pos位置插入元素)
      • [5.6 erase-pos位置删除元素](#5.6 erase-pos位置删除元素)
    • [6 迭代器失效问题](#6 迭代器失效问题)

list的底层结构是双向循环链表,在任意位置插入和删除效率很高,但是不支持任意位置的随机访问。

下面来介绍一些常用的list接口:

1 构造

1.1 无参构造

list()

cpp 复制代码
list<int> lt;

1.2 构造的list中包含n个值为val的元素

list (size_type n, const value_type& val = value_type())

cpp 复制代码
list<int> lt(6, 77);

1.3 用[first, last)区间中的元素构造list

只要是迭代器都可以在其范围内初始化它的内容

list (InputIterator first, InputIterator last)

cpp 复制代码
	list<int> lt1{ 1,2,3,4,5 };
	list<int> lt2(lt1.begin(), lt1.end());

1.4 拷贝构造

list (const list& x)

cpp 复制代码
	list<int> lt1{ 1,2,3,4,5,6,7 };
	list<int> lt2(lt1);

2 迭代器的使用

2.1 begin + end

返回第一个元素的迭代器+返回最后一个元素下一个位置的迭代器

cpp 复制代码
list<int> lt{ 1,2,3,4,5,6,7,8,9,10 };
list<int>::iterator it = lt.begin();
while (it != lt.end())
{
	cout << *it << " ";
	++it;
}
cout << endl;

2.2 rbegin + rend

返回第一个元素的reverse_iterator,即end位置,返回最后一个元素下一个位置的reverse_iterator,即begin位置

cpp 复制代码
list<int> lt{ 1,22,3,44,5,6,67,8,91,10 };
list<int>::reverse_iterator it = lt.rbegin();
while (it != lt.rend())
{
	cout << *it << " ";
	++it;
}
cout << endl;

3 容量操作

list不需要扩容,它的每个节点是独立的一块空间,根据需求来进行新增节点或者删除节点。

3.1 empty + size

empty :检测list是否为空,是返回true,否则返回false
size:返回list中有效节点的个数

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
cout << lt.empty() << endl;//0
cout << lt.size() << endl;//5

4 获取元素

4.1 front + back

返回list第一个节点的值和最后一个节点的值

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
cout << lt.front() << endl;//1
cout << lt.back() << endl;//5

5 插入、删除、修改

5.1 头插-push_front和尾插-push_back

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
lt.push_front(100);
lt.push_back(1000);
for (auto e : lt)
{
	cout << e << " ";
}
cout << endl;

5.2 头删-pop_front和尾删-pop_back

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
lt.pop_front();
lt.pop_back();
for (auto e : lt)
{
	cout << e << " ";
}
cout << endl;

5.3 交换------swap

cpp 复制代码
	list<int> lt1{ 1,2,3,4,5 };
	list<int> lt2{ 6,7,8,9,10 };
	lt1.swap(lt2);

5.4 清理------clear

清空有效元素,保留哨兵位节点

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
lt.clear();
lt.push_back(1);
for (auto e : lt)
{
	cout << e << " ";
}
cout << endl;

5.5 insert-pos位置插入元素

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
auto pos = find(lt.begin(), lt.end(), 3);
lt.insert(pos, 100);
for (auto e : lt)
{
	cout << e << " ";
}
cout << endl;

5.6 erase-pos位置删除元素

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
auto pos = find(lt.begin(), lt.end(), 3);
lt.erase(pos);
for (auto e : lt)
{
	cout << e << " ";
}
cout << endl;

6 迭代器失效问题

前面的例子中都只是使用一次,如果重复使用可能会出现迭代器失效问题,先来insert。

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
auto pos = find(lt.begin(), lt.end(), 3);
lt.insert(pos, 100);
lt.insert(pos, 999);

按正常的思路,第一次在3的前面插入100,第二次在100前面插入999,但是:

list的insert不会出现迭代器失效的问题,但是会出现以上的情况,因为pos指向元素为3的节点,第一次插入是3的前面没有问题,但是第二次插入时pos还是指向元素3,所以插入999还是在元素3的前面插入。

恢复正常逻辑,只需要给pos重新赋值即可。

注:insert和erase是可以有返回值的,返回的是迭代器

cpp 复制代码
pos = lt.insert(pos, 100);
lt.insert(pos, 999);

erase重复使用就会出现因迭代器失效而报错的问题了

先看代码:

cpp 复制代码
list<int> lt{ 1,2,3,4,5 };
auto pos = find(lt.begin(), lt.end(), 3);
lt.erase(pos);
lt.erase(pos);

因为第一次删除pos位置的节点,该节点已经不存在了,第二次删除还是使用上一次的pos,所以会报错。解决方法就是给pos重新赋值

cpp 复制代码
pos = lt.erase(pos);//下一次如果使用,必须重新赋值
lt.erase(pos);//最后一次使用赋不赋值没有关系
相关推荐
Camellia03117 分钟前
嵌入式学习--江协stm32day5
stm32·嵌入式硬件·学习
HUN金克斯8 分钟前
C++/C函数
c语言·开发语言·c++
慢半拍iii9 分钟前
数据结构——F/图
c语言·开发语言·数据结构·c++
钢铁男儿11 分钟前
C# 表达式和运算符(表达式和字面量)
开发语言·c#
a_1571532498612 分钟前
SpringCloud学习笔记-4
笔记·学习·spring cloud
编程有点难14 分钟前
Python训练打卡Day43
开发语言·python·深度学习
m0_6371469320 分钟前
零基础入门 C 语言基础知识(含面试题):结构体、联合体、枚举、链表、环形队列、指针全解析!
c语言·开发语言·链表
LjQ204029 分钟前
网络爬虫一课一得
开发语言·数据库·python·网络爬虫
你是狒狒吗36 分钟前
TM中,return new TransactionManagerImpl(raf, fc);为什么返回是new了一个新的实例
java·开发语言·数据库