c++11基于范围的for循环简化容器遍历,语法为for(decl:expr),支持标准容器、原生数组等;需注意不可修改容器结构、不提供索引、类型推导需谨慎。

C++11 引入了基于范围的 for 循环(range-based for loop),让遍历容器或数组变得更简洁、安全、易读。它自动处理迭代器边界,避免手写 begin()/end() 或下标越界风险,是现代 C++ 中推荐的遍历方式。
语法结构与基本用法
基本形式为:for (declaration : expression) statement
其中 expression 是一个可迭代对象(如 std::vector、std::Array、原生数组、std::String 等),declaration 定义每次迭代的元素变量。
常见写法示例:
-
for (int x : vec) { /* 使用 x 的副本 */ }—— 按值访问,适合小类型或只读场景 -
for (const int& x : vec) { /* 只读引用,避免拷贝 */ }—— 推荐用于大对象或只读遍历 -
for (int& x : vec) { x *= 2; }—— 非 const 引用,允许修改原容器中的元素
支持哪些类型?
只要类型满足以下任一条件,就可使用基于范围的 for:
- 有
begin()和end()成员函数(如标准容器) - 有对应的非成员
begin()/end()函数(可通过 ADL 查找,常用于自定义类型) - 是原生数组(编译器自动推导大小和指针范围)
- 是 C 风格字符串字面量(
"hello"),但注意末尾的' '也会被遍历到
不支持普通指针(如 int* p = new int[5]; for (int x : p) ... 会编译失败),因为指针本身没有 begin/end。
立即学习“C++免费学习笔记(深入)”;
注意事项与常见陷阱
看似简单,但几个细节容易出错:
- 循环中不能直接增删容器元素(如
vec.push_back()或erase()),可能使迭代器失效,引发未定义行为 - 对
std::map或std::unordered_map遍历时,每个元素类型是std::pair<const key value></const>,需按auto& p : myMap获取,再用p.first/p.second - 若想同时获取索引,基于范围的 for 本身不提供;需配合外部计数器,或改用传统 for + 迭代器
- 声明中用
auto很方便,但要注意类型推导规则:例如auto x是值拷贝,auto& x是引用,const auto& x是只读引用——选错可能导致意外拷贝或无法编译
和传统 for 循环怎么选?
优先用基于范围的 for,除非你需要:
- 访问当前索引(如
i) - 反向遍历(可用
rbegin()/rend()配合基于范围的 for,但不如传统 for 直观) - 跳过元素或动态调整步长(如每次加 2)
- 在遍历中修改容器结构(必须用显式迭代器并小心管理)
多数只读或逐个修改的线性遍历场景,基于范围的 for 更清晰、更安全、更不易出错。