1.引入原因
我们可以对算法传递可调用对象,例如find_if
,但是算法只能接受一元谓词或者二元谓词,但是有时候我们希望可调用对象的参数个数会超过算法的要求,这个时候我们可以使用lambda
表达式的捕获列表来添加我们的参数。
eg:find_if
接受的是一元谓词,我们可以通过lambda的捕获列表多操作一个参数。如下代码,通过捕获列表,多操作了一个sz
参数。
int sz = 5;
vector<string> words = { "hello", "world", "interesting", "haha" };
find_if( words.begin(), words.end(), [sz](cosnt string& str) {return str.size() >= sz} );
2.值捕获与引用捕获
- 捕获和函数参数传递一样分值传递和引用传递
- 值传递中,被捕获的值实在lambda创建时,拷贝的。
- 引用捕获和函数引用参数一样是操作原来的对象。
int data = 6;
auto f = [data] {return data; }; // 1
//auto f = [data&] {return data; }; // 2
data = 8;
auto j = f(); //j = 6 in 1, j = 8 in 2
note: 当使用引用捕获时,必须保证lambda
表达式在执行时,被引用的变量是存在的。如果没有必要的话,尽量使用值捕获。
3. 隐式捕获
- 使用
& ,=
来作为捕获列表参数,这样的话,是让编译器根据函数体中使用的变量去推断要捕获的参数,&
是引用捕获,=
是值捕获。
4.返回参数
- 如果函数体不是一句单一的
return
语句,如果没有返回值的话,编译器推断返回void
;如果有返回值的话,那么应该显式指定返回类型,否则会编译错误
auto f = [](int i)->int {if (i < 0) return -i; else return i; };