Lambda 表达式详解

简 述: 详解 Lambda 表达式的基础使用,原理,和使用场景。


本文初发于 “偕臧的小站”,同步转载于此。


Lambda 表达式

语法

[captures](params) mutable exception -> ret {body}

• 捕获表列,捕获当前作用域内的变量,用逗号分隔

• 参数列表

• 可选限定符,可选,mutable允许在函数体内改变按值捕获的变量

• 异常说明符,可选,noexcept

• 返回值类型,可选,多数情况可由编译器自动推导

• 函数体


捕获列表

  • [var] 按值捕获,将 var 按值复制到 Lambda 作用域,默认不能修改

  • [&var] 按引用捕获,将 var 的引用捕获到 Lambda 作用域

  • [=] 捕获所在作用域内全部变量的值

  • [&] 捕获所在作用域内全部变量的引用

  • [this] 在类的成员函数中,捕获当前的 this 指针


初始化捕获

初始化捕获:C++14 支持,在捕获变量时,可以

• 指定闭包类中数据成员的名字

• 直接捕获表达式的结果


原理

• Lambda 表达式:

​ 右值表达式,是源码的一部分,指示编译器生成对应的闭包类和闭包对象

• 闭包类(clousure class):

​ 由编译器根据 Lambda 表达式在编译时生成,每个 Lambda 表达式都有对应的唯一闭包类

• 闭包(clousure):

​ 在运行时创建的闭包类实例,对 Lambda 表达式的调用也就是调用这个实例的成员函数


分为 闭包类 – 捕获变量闭包类 – 不捕获变量 两种;


Lambda 表达式的使用场景

• 创建变量存储 Lambda 表达式

• Lambda 表达式作为函数参数

• Lambda 表达式作为类数据成员

• 使用容器存储 Lambda 表达式


使用 auto 和 decltype 定义 Lambda 表达式变量
auto f = [y](int x){return x * y;}; //右值初始化
auto f2 = f; //拷贝
decltype(f) f3 = std::move(f); //移动
auto&& f4 = [y](int x){return x * y;}; //右值引用绑定右值
const auto& f5 = f4; //常量左值绑定

使用模板将 Lambda 表达式类型参数化
template<typename F>
void fn1(F fn){
    fn();
}
template<typename F>
void fn2(F&& fn){
    fn(std::forward<F>(fn));
}

std::function 函数对象包装器
  • 可以包装任意类型函数对象

  • 函数签名相同的对象可以存储为同一类型

    std::function<float(float, float)> fn;
    fn = std::fmaxf; //普通函数
    fn = std::multiplies<float>(); //仿函数
    fn = [x](float a, float b){return a*b;}; // Lambda 
    

性能问题
  • 动态分配内存

  • 虚函数调用


指针
  • 悬挂引用

    auto make_lambda(int i){
        return [&](int v){
            return v + i;
        };
    }
    auto l = make_lambda(10);
    l(20);
    
  • 悬挂指针

    void Object::f(int i){
        std::thread([=]{
            print(i + m_value);
        }).detach();
    }
    

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

偕臧x

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值