当前位置: 首页 > news >正文

【C++】list的使用

目录

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

list的底层结构是双向循环链表,在任意位置插入和删除效率很高,但是不支持任意位置的随机访问。
在这里插入图片描述
下面来介绍一些常用的list接口:

1 构造

1.1 无参构造

list()

list<int> lt;

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

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

list<int> lt(6, 77);

在这里插入图片描述

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

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

list (InputIterator first, InputIterator last)

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

在这里插入图片描述

1.4 拷贝构造

list (const list& x)

	list<int> lt1{ 1,2,3,4,5,6,7 };list<int> lt2(lt1);

在这里插入图片描述

2 迭代器的使用

2.1 begin + end

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

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位置

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中有效节点的个数

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

4 获取元素

4.1 front + back

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

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

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

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

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

在这里插入图片描述

5.4 清理——clear

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

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位置插入元素

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位置删除元素

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。

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是可以有返回值的,返回的是迭代器
在这里插入图片描述
在这里插入图片描述

pos = lt.insert(pos, 100);
lt.insert(pos, 999);

在这里插入图片描述

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

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重新赋值

pos = lt.erase(pos);//下一次如果使用,必须重新赋值
lt.erase(pos);//最后一次使用赋不赋值没有关系

在这里插入图片描述

http://www.lryc.cn/news/286396.html

相关文章:

  • mybatis的缓存机制
  • ChatGLM3报错:No chat template is defined for this tokenizer
  • 大数据学习之Flink、搞懂Flink的恢复策略
  • C语言易忘操作符全集
  • 网络请求 mvp mvvm get post delete put 请求
  • 研究生开题报告撰写:文言一心VSChatgpt3.5
  • Unity animator动画倒放的方法
  • Dubbo源码解析第一期:如何使用Netty4构建RPC
  • unity刷新grid,列表
  • 蓝桥杯备赛 day 3 —— 高精度(C/C++,零基础,配图)
  • 人形机器人创新发展顶层设计与关键技术布局
  • C语言-算法-最小生成树
  • android 扫描某个包下的所有类
  • 远程ssh 不通的原因之一
  • wamp集成环境部署
  • 使用antd design pro 及后端nodejs express 结合minio进行文件的上传和下载管理
  • Unity常用的优化技巧集锦
  • c++动态调用dll
  • 使用Python自动化操作手机,自动执行常见任务,例如滑动手势、呼叫、发送短信等等
  • E - Souvenir(图论典型例题)
  • 【SpringBoot篇】添加富文本编辑器操作
  • 前台vue配置
  • 牛客周赛 Round 18 解题报告 | 珂学家 | 分类讨论计数 + 状态DP
  • CentOS防火墙基本操作
  • Shell脚本的编程规范和变量类型
  • C++面试:跳表
  • 掌握C++20的革命性特性:Concepts
  • win11开机后频繁刷新桌面,任务栏不显示,文件资源管理器explorer报错ntdll.dll
  • 解决Git添加.gitignore文件后不生效的问题
  • Shell Linux学习笔记