c++中如何使用std::for_each遍历容器_c++迭代器算法用法【实例】

std::for_each是中的泛型算法,本质为函数调用而非循环语法糖,接收迭代器区间和可调用对象,对每个元素执行一次,无break/continue、不返回中间状态,强调副作用明确的扁平逻辑。

std::for_each 本质是函数调用,不是循环语法糖

std::for_each 中的泛型算法,它接收一对迭代器和一个可调用对象,对区间内每个元素执行该可调用对象。它不提供“跳出”或“跳过”的能力(不像 breakcontinue),也不返回中间状态 —— 执行完就结束,返回传入的函数对象(常被忽略)。

常见误解是把它当 for 循环的替代写法,但实际它强调「对每个元素做一件事」的语义,适合副作用明确、逻辑扁平的场景(如打印、更新、收集日志)。若需条件中断、索引访问或提前退出,应优先考虑传统 forwhile 循环。

正确传参:迭代器范围必须合法,函数对象要能接受解引用类型

传给 std::for_each 的两个迭代器必须构成有效

左闭右开区间(first 可达 last,且 last 不可解引用)。函数对象参数的形参类型必须与容器元素类型兼容(自动推导,但常因 const/volatile/引用修饰出错)。

  • std::vector,lambda 形参用 intconst int&int& 都合法;用 int&& 会编译失败
  • std::map,元素类型是 std::pair,不能直接用 auto& key 拆包 —— 必须解构 pair 或用结构化绑定(C++17+)
  • 若容器为空,first == laststd::for_each 不执行任何调用,安全

常见错误:捕获变量生命周期、修改容器导致迭代器失效

在 lambda 中通过引用捕获局部变量时,必须确保 lambda 的生命周期不超过该变量;更危险的是在遍历过程中修改容器大小(如 push_backerase),这会使所有现存迭代器失效,std::for_each 行为未定义。

以下示例看似自然,实则危险:

std::vector v = {1, 2, 3};
std::for_each(v.begin(), v.end(), [&v](int x) {
    if (x == 2) v.push_back(4); // ❌ 迭代器失效!
});

正确做法是:先收集待操作项,再统一处理;或改用支持安全擦除的模式(如 remove_if + erase)。

C++11 以后推荐用范围 for,std::for_each 更适合配合 std::bind 或函数对象复用

对简单遍历,for (const auto& x : container) 更直观、不易出错、支持 break/continue。而 std::for_each 的优势在于:可与 std::bind 组合复用逻辑,或作为高阶函数参数传递。

例如,封装一个通用打印器:

#include 
#include 
#include 
#include 

void print_with_prefix(const std::string& prefix, int x) {
    std::cout << prefix << x << '\n';
}

int main() {
    std::vector v = {10, 20, 30};
    auto printer = std::bind(print_with_prefix, "[LOG] ", _1);
    std::for_each(v.begin(), v.end(), printer); // 输出三行 [LOG] 10 等
}

真正需要 std::for_each 的地方,往往不是“怎么遍历”,而是“如何把遍历动作本身参数化或延迟执行”。否则,别硬套。