值捕获复制变量,引用捕获共享变量;前者安全独立,后者实时同步但需防悬空引用。

在C++中,lambda表达式是一种定义匿名函数的简洁方式,广泛用于STL算法、回调函数等场景。捕获列表(capture clause)是lambda表达式的重要组成部分,决定了lambda如何访问其定义作用域中的变量。理解值捕获与引用捕获的区别,对正确使用lambda至关重要。
捕获列表的基本语法
lambda表达式的完整形式如下:
[capture](parameters) -> return_type { body }其中 capture 就是捕获列表,用来“捕获”外部作用域的变量,使其在lambda内部可用。常见的捕获方式包括:
- []:不捕获任何变量
- [=]:以值的方式捕获所有外部变量
- [&]:以引用的方式捕获所有外部变量
- [var]:值捕获特定变量
- [&var]:引用捕获特定变量
- [this]:捕获当前对象的指针
- [=, &var]:混合捕获,值捕获为主,个别引用捕获
- [&, var]:引用捕获为主,个别值捕获
值捕获 vs 引用捕获的区别
关键区别在于生命周期和数据同步:
立即学习“C++免费学习笔记(深入)”;
值捕获(by value)
- 在lambda创建时,将外部变量的副本保存到lambda的闭包对象中
- 后续外部变量的变化不会影响lambda内部的值
- 适用于变量生命周期短于lambda的情况
- 注意:修改被捕获的值需要加上 mutable 关键字
引用捕获(by reference)
- lambda内部保存的是外部变量的引用
- lambda执行时访问的是变量的当前值,因此能反映外部变化
- 风险:若外部变量已销毁而lambda仍被调用,会导致未定义行为
- 不需要 mutable 即可修改原变量
实际示例对比
int x = 10;auto val_lambda = [x]() { return x; };
auto ref_lambda = [&x]() { return x; };
x = 20;
std::cout
std::cout
可见,值捕获的lambda返回的是创建时的副本,而引用捕获反映最新值。
何时使用哪种捕获方式?
优先使用值捕获的场景:- lambda可能在原作用域结束后才被调用(如异步任务)
- 希望lambda有独立的数据状态
- 捕获基本类型(int、double等),开销小
使用引用捕获的场景:
- 需要修改外部变量
- 捕获大型对象避免拷贝开销
- lambda与外部代码紧密协作,需实时同步状态
但要确保lambda的生命周期不超过所引用变量的生命周期。
基本上就这些。掌握捕获列表的行为,能避免常见陷阱,比如悬空引用或意外的数据隔离。写lambda时多思考变量的生命周期和共享需求,选择合适的捕获方式。









