【lambda函数】lambda()函数

简介: 【lambda函数】lambda()函数

lambda()语法

lambda表达式书写格式:

[capture-list] (parameters) mutable -> return-type
 { 
   statement
}

咱们一个个来解释:

[capture-list] :捕捉列表,该列表总是出现在lambda函数的开始位置,编译器根据 [] 来判断接下来的代码是否为lambda函数,捕捉列表能够捕捉上下文中的变量供lambda函数使用。不能省略。

(parameters): 参数列表,与普通函数的参数列表一致,如果不需要参数传递,则可以连同()一起省略

mutable: 默认情况下,lambda函数总是一个const函数,mutable可以取消其常量性。使用该修饰符时,参数列表不可省略(即使参数为空)。

-> return-type:返回值类型,用追踪返回类型形式声明函数的返回值类型,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。

statement:函数体,在该函数体内,除了可以使用其参数外,还可以使用所有捕获到的变量。不可省略。

因此,把可以省略的都省略掉,最那简单的lambda函数是 []{} ,该 lambda 表达式没有任何意义。该lambda函数不能做任何事情。

接下来写一个lambda函数:

auto add = [](int x, int y)->int
    {
        return x + y;
    };

lambda表达式实际上可以理解为无名函数,该函数无法直接调用,如果想要直接调用,可借助auto将其赋值给一个变量。

调用:

  //第一种
    add(10, 20);
    //第二种
    [](int x, int y)->int {return x + y;}(10, 20);

可以看出,lambda函数和普通函数在组成和调用上都很相似。参数列表,返回值,函数体都不在多叙述。

捕捉列表

捕捉列表描述了上下文中那些数据可以被lambda使用。

如:

    int a = 10;
    int b = 20;
    auto add = [a,b]() 
    {
        return a + b ;
    };

直接捕捉了 a b 变量,且是传值捕捉,lambda函数体内的a, b变量,只是外边 a b 的一份拷贝。且默认无法修改。

要想修改,可以使用 mutable 进行修饰。

mutable

如:

auto add = [a, b]() mutable
    {
        a = 20;
        return a + b;
    };

就不会报错,但因为是传值,所以lambda 函数内部 a的变化,无法影响外部的a变量。

mutable 用的比较少。

当然,lambda函数 和普通函数一样,捕捉列表,可以传值捕捉,也可以传引用捕捉。

    int x = 10;
  int y = 20;
  //捕捉列表
  //传引用   参数列表
  auto fun1 = [](int& x, int& y) {
    int tmp = x;
    x = y;
    y = x;
  };
    // 传引用捕捉   
  auto fun2 = [&x,&y]() {
    int tmp = x;
    x = y;
    y = x;
  };
    //对上下文所有变量进行传引用捕捉
  auto fun3 = [&]() {
    x = y;
  };
  //对除y以外的所有变量传引用捕捉,y传值捕捉
  auto fun4 = [&, y] {
    ;
  };
    //对y进行传值捕捉,对其余变量进行传引用捕捉
  auto fun5 = [=, &y] {
    ;
  };

lambda 底层原理

看如下代码:

int main()
{
    int a = 10;
    int b = 20;
    auto add = [a, b]() mutable
    {
        a = 20;
        return a + b;
    };
    cout << typeid(add).name() << endl;
    cout << sizeof(add) << endl;
  return 0;
}

lambda 函数的类型变量是什么呢?

lambda 类型的大小又是多少呢?

从运行结果上来看,其大小为一,类型大致为一个类,具体是什么我们现在也不清楚。

函数对象与lambda表达式

函数对象,又称为仿函数,即可以想函数一样使用的对象,就是在类中重载了operator()运算符的类对象。

//仿函数
class math {
public:
    int operator() (int x, int y)
    {
        return x + y;
    }
};
int main()
{
    int a = 10;
    int b = 20;
    //仿函数对象
    math m;
    //lambda函数
    auto add = [](int a,int b) 
    {
        return a + b;
    };
    m(a, b);
    add(a, b);
  return 0;
}

我们从反汇编上来看,

仿函数底层代码,调用了 重载的 ()

lambda() 函数的底层:

我们也可以看出,也是调用了一个lambda 类里的重载的(),

不妨看出,lambda()函数的底层就是一个重载了()的空类。

所以就可以知道,lambda类型的大小为1了:

因为,它的底层是一个空类,是一个仿函数。

至于它的类型,如图:

也就是,上图是 lambda_UUID

UUID 是 通用唯一识别码(Universally Unique Identifier)的缩写,是通过一种特殊的算法计算出来的具有唯一识别信息的 数据。

也就是说,每一个lambda()对象的类型都不一样。

也就不存在不同lambda()对象相互赋值的情况。

结语

本次的博客就到这了。

我是Tom-猫,

如果觉得有帮助的话,记得

一键三连哦ヾ(≧▽≦*)o。

咱们下期再见。

相关文章
|
7月前
|
存储 JavaScript 前端开发
c++lambda函数与表达式
c++lambda函数与表达式
39 1
|
Python
lambda
lambda 是一个匿名函数,它通常用于简化代码,使代码更简洁、易读。lambda 函数不需要显式地使用 def 关键字进行定义,可以直接在表达式中使用。它的语法如下:
176 6
|
SQL Java 程序员
聊聊lambda
聊聊lambda
48 1
|
7月前
|
C#
C# Lambda
C# Lambda
45 0
|
设计模式 Java API
值得使用Lambda的8个场景,别再排斥它了!
前言 可能对不少人来说,Lambda显得陌生又复杂,觉得Lambda会导致代码可读性下降,诟病Lambda语法,甚至排斥。
|
算法 编译器 容器
lambda
lambda
110 0
|
Java 开发者
lambda让代码更优雅
Lambda表达式是Java 8中引入的一个重要特性,它允许开发者以更简洁的方式编写匿名函数,使得代码更加紧凑和易读。Lambda表达式是函数式编程的一种体现,可以将函数作为方法的参数传递,并且可以使用更简洁的语法实现函数式接口(只有一个抽象方法的接口)的实例化。Lambda表达式的语法形式为 (参数列表) -> {表达式或语句块}。
86 0
|
Serverless Go 云计算
Lambda
Lambda自制脑图
238 0
 Lambda