news 2026/5/1 5:03:04

c++遍历容器(vector、list、set、map

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
c++遍历容器(vector、list、set、map

遍历vector

1.基于范围的 for 循环(C++11 起推荐使用)

#include <vector> #include <iostream> std::vector<int> vec = {1, 2, 3, 4, 5}; for (const auto& element : vec) { std::cout << element << " "; }
  • 使用const auto&避免不必要的拷贝(尤其对大型对象有用)。
  • 如果你需要修改元素,可以去掉const或使用auto&

2.使用迭代器

for (auto it = vec.begin(); it != vec.end(); ++it) { std::cout << *it << " "; }

或者使用const_iterator(只读):

for (auto it = vec.cbegin(); it != vec.cend(); ++it) { std::cout << *it << " "; }

3.传统下标循环

for (size_t i = 0; i < vec.size(); ++i) { std::cout << vec[i] << " "; }
  • 注意:size()返回的是size_t(无符号类型),避免与有符号整数比较产生警告。

4.使用 std::for_each(函数式风格)

#include <algorithm> std::for_each(vec.begin(), vec.end(), [](int n) { std::cout << n << " "; });

输出两层循环的vector

方法 1:基于范围的 for 循环(推荐)

for (const auto& row : matrix) { for (const auto& elem : row) { std::cout << elem << " "; } std::cout << "\n"; // 每行结束后换行 }

方法 2:传统下标循环

for (size_t i = 0; i < matrix.size(); ++i) { for (size_t j = 0; j < matrix[i].size(); ++j) { std::cout << matrix[i][j] << " "; } std::cout << "\n"; }

⚠️ 注意:使用size_t避免有符号/无符号比较警告。

方法 3:迭代器方式

for (auto row_it = matrix.begin(); row_it != matrix.end(); ++row_it) { for (auto elem_it = row_it->begin(); elem_it != row_it->end(); ++elem_it) { std::cout << *elem_it << " "; } std::cout << "\n"; }

遍历deque

在 C++ 中,std::deque(双端队列)是一个支持在两端高效插入/删除、同时支持随机访问的顺序容器。遍历deque的方式非常灵活,几乎和vector一样丰富。

方法 1:基于范围的 for 循环(C++11 起,推荐)

#include <iostream> #include <deque> std::deque<int> dq = {10, 20, 30, 40}; for (const auto& elem : dq) { std::cout << elem << " "; } // 输出: 10 20 30 40

如果需要修改元素:

for (auto& elem : dq) { elem *= 2; // 所有元素翻倍 }

方法 2:使用下标(随机访问)

for (size_t i = 0; i < dq.size(); ++i) { std::cout << dq[i] << " "; // 或 dq.at(i) }

方法 3:使用迭代器(正向)

// 普通迭代器 for (auto it = dq.begin(); it != dq.end(); ++it) { std::cout << *it << " "; } // 只读迭代器(推荐用于只读场景) for (auto it = dq.cbegin(); it != dq.cend(); ++it) { std::cout << *it << " "; }

方法 4:反向遍历(从尾到头)

// 基于范围(C++17 不支持直接反向范围 for,需用迭代器) for (auto rit = dq.rbegin(); rit != dq.rend(); ++rit) { std::cout << *rit << " "; } // 输出: 40 30 20 10

方法 5:使用 std::for_each + lambda

#include <algorithm> std::for_each(dq.begin(), dq.end(), [](int n) { std::cout << n << " "; });

⚠️ deque 遍历注意事项

特性说明
✅ 支持随机访问可用dq[i]dq.at(i),时间复杂度 O(1)
✅ 支持双向迭代++it/--it都高效
✅ 迭代器可能失效仅当在头部或尾部扩容时,所有迭代器可能失效(与vector类似,但比list更敏感)
🔄 内存非连续虽然支持随机访问,但底层是分段连续存储,缓存性能略低于vector

遍历set

在 C++ 中,std::set是一个有序、唯一的关联容器(通常基于红黑树实现),遍历方式与vector类似,但由于其元素是const 的(不可修改),有一些细节需要注意。

1.基于范围的 for 循环(推荐,C++11 起)

#include <iostream> #include <set> std::set<int> mySet = {3, 1, 4, 1, 5}; for (const auto& elem : mySet) { std::cout << elem << " "; // 输出: 1 3 4 5 }
  • 元素自动按升序排列(默认使用std::less)。
  • 不能通过遍历修改元素(因为set中的 key 是 const 的),所以用const auto&是安全且高效的。

2.使用迭代器

for (auto it = mySet.begin(); it != mySet.end(); ++it) { std::cout << *it << " "; }

或只读迭代器(更明确):

for (auto it = mySet.cbegin(); it != mySet.cend(); ++it) { std::cout << *it << " "; }

3.使用 std::for_each + lambda

#include <algorithm> std::for_each(mySet.begin(), mySet.end(), [](const int& n) { std::cout << n << " "; });
  • std::set不支持下标访问(如mySet[i]是非法的)。
  • 遍历时元素自动排序(除非你自定义了比较函数)。
  • 所有遍历方式都是只读的——这是由set的设计决定的(修改 key 会破坏内部结构)。

遍历list

在 C++ 中,std::list是一个双向链表容器,支持高效的插入和删除操作(但不支持随机访问)。遍历std::list的方式与其他 STL 容器类似,但由于其底层是链表结构,不能使用下标(如list[i]

注意事项

特性说明
❌ 不支持下标访问mylist[0]是非法的,编译会报错
✅ 支持修改元素通过非 const 引用或迭代器可修改值
✅ 迭代器稳定插入/删除(除被删元素外)不会使其他迭代器失效
🔄 双向遍历支持++it--it,以及rbegin()/rend()

方法 1:基于范围的 for 循环(C++11 起,推荐)

#include <iostream> #include <list> std::list<int> mylist = {10, 20, 30, 40}; for (const auto& elem : mylist) { std::cout << elem << " "; } // 输出: 10 20 30 40

如果需要修改元素,可用auto&

for (auto& elem : mylist) { elem *= 2; // 所有元素翻倍 }

方法 2:使用迭代器

// 正向遍历 for (auto it = mylist.begin(); it != mylist.end(); ++it) { std::cout << *it << " "; } // 只读遍历(更安全) for (auto it = mylist.cbegin(); it != mylist.cend(); ++it) { std::cout << *it << " "; }

方法 3:反向遍历(从尾到头)

for (auto rit = mylist.rbegin(); rit != mylist.rend(); ++rit) { std::cout << *rit << " "; } // 输出: 40 30 20 10

方法 4:使用 std::for_each + lambda

#include <algorithm> std::for_each(mylist.begin(), mylist.end(), [](int n) { std::cout << n << " "; });

性能提示

  • std::list遍历速度通常比std::vector(因为内存不连续,缓存不友好)。
  • 如果只是顺序读取且不需要频繁中间插入/删除,优先考虑vectordeque

遍历map

特性说明
自动排序std::map内部基于红黑树,遍历时按键升序输出
key 不可修改遍历时pair.firstconst,不能修改 key
value 可修改可通过pair.second = newValue修改值(若用非 const 引用)
无重复 key插入相同 key 会覆盖旧值

在 C++ 中,std::map是一个有序的键值对(key-value)关联容器,每个元素是一个std::pair<const Key, Value>。遍历map的结果会按键的升序自动排序(默认使用std::less<Key>)。

方法 1:基于范围的 for 循环(C++11 起,推荐)

#include <iostream> #include <map> #include <string> std::map<std::string, int> scores = { {"Alice", 95}, {"Bob", 87}, {"Charlie", 92} }; for (const auto& pair : scores) { std::cout << pair.first << " => " << pair.second << "\n"; }

或者使用结构化绑定(C++17 起更清晰):

for (const auto& [key, value] : scores) { std::cout << key << " => " << value << "\n"; }

方法 2:使用迭代器

for (auto it = scores.begin(); it != scores.end(); ++it) { std::cout << it->first << " => " << it->second << "\n"; }

方法 3:使用 std::for_each + lambda

#include <algorithm> std::for_each(scores.begin(), scores.end(), [](const auto& p) { std::cout << p.first << " => " << p.second << "\n"; });

一、基本特性总览

容器底层结构是否有序元素是否唯一支持随机访问插入/删除效率(典型位置)是否支持下标[]
vector动态数组✅ O(1)尾部:O(1)
中间/头部:O(n)
deque分段连续数组✅ O(1)两端:O(1)
中间:O(n)
list双向链表任意位置:O(1)(需迭代到位置)
set红黑树(平衡 BST)✅(按键)任意位置:O(log n)
map红黑树✅(按键)✅(key 唯一)任意位置:O(log n)✅(map[key],但会插入默认值)

遍历方式对比

容器范围 for下标[]迭代器反向遍历遍历时修改元素
vector✅(非 const)
deque
list
set❌(key 是 const)
map❌(不能用于遍历)✅(可改 value,不能改 key)
性能对比(时间复杂度)
操作vectordequelistset/map
尾部插入O(1) amortizedO(1)O(1)O(log n)
头部插入O(n)O(1)O(1)O(log n)
中间插入O(n)O(n)O(1)*O(log n)
随机访问O(1)O(1)O(n)
查找(按值)O(n)O(n)O(n)O(log n)
删除(已知位置)O(n)O(n)O(1)O(log n)
内存局部性⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐

使用场景建议

容器推荐使用场景
vector- 需要频繁随机访问
- 数据量固定或主要在尾部增删
- 对缓存友好(高性能循环)
deque- 需要在两端高效插入/删除(如队列、滑动窗口)
- 不需要中间插入
list- 需要在任意位置频繁插入/删除(且已有迭代器)
- 不关心随机访问和内存连续性
- 实现 LRU 缓存等
set- 需要自动去重 + 自动排序
- 快速判断元素是否存在
map- 需要键值映射 + 自动按键排序
- 快速通过 key 查找 value

完整容器列表

容器类型是否有序允许重复底层实现头文件
std::set✅ 是❌ 否红黑树(平衡二叉搜索树)<set>
std::multiset✅ 是✅ 是红黑树<set>
std::unordered_set❌ 否❌ 否哈希表 + 链地址法(或开放寻址)<unordered_set>
std::unordered_multiset❌ 否✅ 是哈希表<unordered_set>
std::map✅ 是❌ 否(key 唯一)红黑树<map>
std::multimap✅ 是✅ 是(相同 key 可多次出现)红黑树<map>
std::unordered_map❌ 否❌ 否(key 唯一)哈希表<unordered_map>
std::unordered_multimap❌ 否✅ 是哈希表<unordered_map>

所有unordered_*容器要求key 类型支持哈希(即提供std::hash<Key>特化),否则需自定义哈希函数。


时间复杂度对比(平均 / 最坏)

操作set/map
(红黑树)
multiset/multimapunordered_*
(哈希表)
插入O(log n)O(log n)平均 O(1)
最坏 O(n)(哈希冲突严重或 rehash)
删除O(log n)O(log n)平均 O(1)
最坏 O(n)
查找(by key)O(log n)O(log n)平均 O(1)
最坏 O(n)
范围查询
(如lower_bound
✅ O(log n)❌(不支持有序范围)
遍历全部元素O(n),有序O(n),有序O(n),无序
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/24 17:58:10

3个技巧让文献管理效率提升200%:Zotero Ethereal Style插件实战指南

3个技巧让文献管理效率提升200%&#xff1a;Zotero Ethereal Style插件实战指南 【免费下载链接】zotero-style zotero-style - 一个 Zotero 插件&#xff0c;提供了一系列功能来增强 Zotero 的用户体验&#xff0c;如阅读进度可视化和标签管理&#xff0c;适合研究人员和学者。…

作者头像 李华
网站建设 2026/4/17 13:20:39

PPT悬浮计时器完全指南:高效掌控演讲时间的秘诀

PPT悬浮计时器完全指南&#xff1a;高效掌控演讲时间的秘诀 【免费下载链接】ppttimer 一个简易的 PPT 计时器 项目地址: https://gitcode.com/gh_mirrors/pp/ppttimer 在竞争激烈的职场环境中&#xff0c;精准的时间管理是演讲成功的关键。数据显示&#xff0c;超过60%…

作者头像 李华
网站建设 2026/4/8 9:37:39

FaceRecon-3D效果对比:与ECCV 2023主流方法在纹理保真度上的实测分析

FaceRecon-3D效果对比&#xff1a;与ECCV 2023主流方法在纹理保真度上的实测分析 1. 这不是“建模软件”&#xff0c;而是一张照片变3D人脸的魔法 你有没有试过&#xff0c;把手机里一张普通自拍上传到某个网页&#xff0c;几秒钟后&#xff0c;屏幕上就出现了一张“摊开的人…

作者头像 李华
网站建设 2026/4/16 15:56:51

Llama-3.2-3B轻量推理:Ollama平台下3B模型在RTX 3060 12GB稳定运行

Llama-3.2-3B轻量推理&#xff1a;Ollama平台下3B模型在RTX 3060 12GB稳定运行 你是否试过在消费级显卡上跑大模型&#xff1f;不是动辄需要48G显存的A100&#xff0c;也不是得插满三块卡的服务器配置——而是一张手头就有的RTX 3060 12GB&#xff0c;安静地放在办公桌下&…

作者头像 李华
网站建设 2026/4/16 15:09:48

ChatTTS未来展望:下一代拟真语音合成的技术路径

ChatTTS未来展望&#xff1a;下一代拟真语音合成的技术路径 1. 它不是在读稿&#xff0c;而是在“活”着说话 你有没有听过一段语音&#xff0c;下意识停顿两秒&#xff0c;然后才反应过来——这居然是AI生成的&#xff1f; 不是那种字正腔圆、节奏工整、像新闻联播一样的“…

作者头像 李华