看下面例子,将容器中的数值都变成绝对值,然后存储到另外一个容器中:
#include <iostream> #include <vector> #include <iterator> #include <algorithm> int main(){ std::vector<int> v1={1,2,3,-4,-5,-6},v2; auto iv=std::back_inserter(v2); std::for_each(v1.begin(),v1.end(),[iv](const int &i) mutable {i>0 ? *iv=i : *iv=-i;}); *iv=999; for(const auto &i:v2) std::cout<<i<<' '; std::cout<<std::endl; return 0; }
代码输出值为:1 2 3 4 5 6 999
上面有两点需要注意:
1. lambda表达式中捕获列表如果是值捕获的话,如果需要改变捕获后的变量或者变量指向的值,则必须在函数体{}前面加上mutable ,不然编译失败;
2. 一般情况下,对于普通的迭代器,如果容器的元素个数改变的话,迭代器很可能会失效,但是对于插入迭代器,此迭代器是不会失效的,lambda表达式中的迭代器iv和外部的iv不是同一个变量。在lambda表达式中,我们用lambda中的iv为容器v2创建元素,并不是用外部的iv来创建的。不过下面的 *iv=999; 是用外部iv来创建元素的。来看一个例子:
std::vector<int> v2; auto iv=std::back_inserter(v2); auto iv1=std::back_inserter(v2); *iv=4; *iv1=5;v2中保存的元素为4, 5
此外,如果将lambda表达式中的捕获列表改为引用捕获,则就不需要mutable就能在lambda中改变被捕获变量的值了:
std::for_each(v1.begin(),v1.end(),[&iv](const int &i) {i>0 ? *iv=i : *iv=-i;});
并且此时lambda表达式中的iv与外部的iv是同一个变量。