跳到正文
Carol's Blog
返回

C++小技巧-Lambda表达式

由于最近项目中用到了C++ Lambda表达式的相关内容,在这里记录一下。

什么是Lambda表达式

我自己理解的是Lambda表达式就是不需要复杂逻辑的匿名函数,在很多高级语言中都有。

比如Python中:

my_list = [1, 2, 3]
new_list = list(map(lambda x: 'new' + str(x), my_list))

就可以在map中使用Lambda表达式使得程序更加简洁。

C++的Lambda表达式

C++11提供了Lambda表达式的功能,完整声明的格式如下:

[capture_list] (params_list) mutable exception -> return_type { function_body }

各项定义如下:

这6部分中,除了capture_listfunction_body,其他4部分都是可选的。

常见的情况有以下三种:

捕获外部变量-进阶说明

在C++中,函数参数的传递方式有三种:值传递、引用传递和指针传递。

Lambda表达式的外部变量捕获方式有三种:值捕获、引用捕获和隐式捕获。

1.值捕获

capture_list中直接传入变量值,在Lambda表达式构建时,外部变量将通过值拷贝的方式传入,如果后面外部变量被修改,也不会影响Lambda表达式中的值

int main() {
    int val = 0;
    auto func = [val] { cout << val << endl; };
    val = 1;
    func(); // 输出 0
}

2.引用捕获

在外部变量名前加&,使用引用捕获的变量,将会在Lambda表达式中与外部变量绑定。在表达式构造结束,该外部变量变化之后再调用Lambda表达式,这时会使用最新的外部变量对象。

int main() {
    int val = 0;
    auto func = [&val] { cout << val << endl; };
    val = 1;
    func(); // 输出 1
}

3.隐式捕获

除了在capture_list中指定外界变量名之外,还可以用隐式捕获的办法,即让Lambda表达式自行推断需要哪些外部变量。隐式捕获具体有两种:

4.混合捕获

C++11中的Lambda表达式支持将以上三种捕获方式混用,常见的情况如下:

捕获形式说明
[]不捕获外部变量
[a, b, ...]全部以值捕获捕捉
[this]捕获this指针
[=]值捕获所有外部变量
[&]引用捕获所有外部变量
[=, &a]变量a使用引用捕获,其他使用值捕获
[&, a]变量a使用值捕获,其他使用引用捕获

捕获外部变量并修改

当外部参数采用值捕获时,在function_body函数体内部无法修改捕获的外部变量值,会提示read-only。 这时需要添加mutable关键字。

int main() {
    int val = 0;
    auto func = [val]() mutable { cout << ++val << endl; };
    func(); // 输出 1
    cout << val << endl; // 输出 0
}

如果使用引用捕获,则不需要添加mutable关键字,在Lambda表达式中更改的内容会直接影响到外部变量。

Lambda表达式参数params_list的限制

Lambda表达式的参数和普通函数的参数类似,不过有一些限制:

  1. 参数列表中不能有默认参数
  2. 不支持可变参数,即用...表示的参数,不定个数。
  3. 所有参数必须有参数名

其他例子

#include <iostream>
#include <functional>

using namespace std;

int main() {
    auto funcA = [](int x) -> function<int(int, int)> { return [=](int y, int z) { return (x + y) * z; }; };

    cout << funcA(2)(4, 8) << endl; // 输出 48      (2 + 4) * 8 = 48

    auto funcB = [](const function<int(int, int)>& func, int valA, int valB) { return -func(valA, valB); };

    cout << funcB(funcA(3), 6, 9) << endl;  // 输出 -81    -(2 + 6) * 9
    return 0;
}}

分享这篇文章:
通过邮件分享这篇文章

分享到微信

微信对普通网页没有开放通用直连分享协议。更稳妥的方式是复制链接、扫码打开,或在支持的设备上调用系统分享。

上一篇
C++-架构之路
下一篇
qqbot插件-闪照转发