Lambda表达式的本质

article/2025/10/4 0:51:44

一直想写一篇文章,来总结lambda表达式,但是之前感觉总结的不是特别到位,现在看了几篇文章和视频后,感觉对lambda表达式有了比较深刻的认识,现在进行记录总结如下:

lambda表达式又叫做匿名函数,lambda表达式本质是一个匿名类,lambda为了我们能更方便的定义出一个函数方法提供了便利,下面咱们开始介绍下lambda表达式。

1.lambda表达式的引出

首先我们想完成一个   Add的函数,这个函数有个功能,就是把自己成员变量的值num_,和传进来的参数x相加,并且返回。我们一般用C++的写法如下:

#include<typeinfo>
#include<iostream>
#include<functional>
using namespace std;
class Cal{
public:
Cal(int num):num_(num){cout<<"num_ :"<<num_<<endl;
};
int Add(int x){ cout<<"x+num_:"<<x+num_<<endl;return x+num_;
};
private:const int num_;
};int main(){Cal temp(5);temp.Add(3);return 0;
}

编译并且执行程序:

zhc@ubuntu:~/test/linux$ g++ -std=c++14  test.cpp -o test && ./test 
num_ :5
x+num_:8

接下来我们用仿函数来实现,仿函数也就是重载小括号  ()的类,我们修改代码如下:

#include<memory>
#include<typeinfo>
#include<iostream>
#include<functional>
using namespace std;
class Cal{
public:
Cal(int num):num_(num){cout<<"num_ :"<<num_<<endl;
};
int operator () (int x){ cout<<"x+num_:"<<x+num_<<endl;return x+num_;
}
private:const int num_;
};
int main(){Cal Add(5);Add(3);return 0;
}

编译并且输出结果:

zhc@ubuntu:~/test/linux$ g++ -std=c++14  test.cpp -o test && ./test 
num_ :5
x+num_:8

也就是说如果我们想实现一个简单的Add函数,也要定义一个类,或者用成员函数实现,或者用仿函数实现,有没有一种写法上更简单的方式呢,省略掉类的定义,那么我们的主角就要登场了,就是lambda表达式.看如下代码:

#include<memory>
#include<typeinfo>
#include<iostream>
#include<functional>
using namespace std;
int main(){int num=5;auto lam_add=[num](int x) ->int {cout<<"num+x:"<<num+x<<endl;return num+x;};  lam_add(3);return 0;
}

是不是形式上简洁了许多,输出结果如下:

zhc@ubuntu:~/test/linux$ g++ -std=c++14  test.cpp -o test && ./test 
num+x:8

正如我们的lambda表达式结果看,这个跟我们刚才写的仿函数类也是基本相同的,实际上我写的第二个例子,就是这个lambda表达式在调用的时候,生成的临时类对象,所对应的类的定义,

注意我的成员变量的写法,const int num_,这个是我故意这么写的,因为确实等价方式就是这样。下面我们来讲一讲lambda表达式的捕获列表,以及不同捕获形式,在生成类的成员函数里的可读可写属性。

2. lambda表达式的捕获列表

lambda表达式的书写形式如下:

[captrue] (params) opt -> ret {body};
其中,capture是捕获列表;params是参数列表;
opt是函数选项(mutable,noexcept等关键字);
ret是返回值类型;body是函数体。

lambda表达式可以通过捕获列表捕获一定范围内的变量:

[]不捕获任何变量;

[bar]按值捕获bar变量,同时不捕获其他变量,并作为副本在函数体使用(按值捕获);

[&bar]按引用捕获bar变量,同时不捕获其他变量;并作为引用在函数体使用(按引用捕获);

[=]捕获外部作用域作用变量,并作为副本在函数体使用(按值捕获);

[&]捕获外部作用域所有变量,并作为引用在函数体使用(按引用捕获);

[=,&foo]按值捕获外部作用域所有变量,并按引用捕获foo变量;

[this]捕获当前类中的this指针,让lambda拥有和当前类成员函数同样的访问权限,如果已经使用了&或者=,就默认添加此选项。捕获this的目的是可以在lambda中使用当前类的成员函数和成员变量。

一般,不会捕获全局变量,静态全局变量,局部静态局部变量,静态成员变量。

下面我们按照按值捕获和按引用捕获来说明下,捕获后在生成的匿名类里的成员变量情况,与外部变量的可写属性是否相同,以及是否能够改变外部变量。

下面咱们看下具体的例子:

#include<memory>
#include<typeinfo>
#include<iostream>
#include<functional>
using namespace std;
int main(void){
int m=1;
const int n=2;
auto g = [m,n](int x)->int{
m++;
n++;
return m+n;};
int z=g(0);
return 0;
}

编译代码并且执行:编译报错如下:

zhc@ubuntu:~/test/linux$ g++ -std=c++14  test.cpp -o test && ./test 
test.cpp: In lambda function:
test.cpp:10:2: error: increment of read-only variable ‘m’m++;^~
test.cpp:11:2: error: increment of read-only variable ‘n’n++;^~

具体结论相关的东西见下面的两个图,说的更清楚一些。

 这个图是我从别的地方copy过来的,暂时粘贴这里,后期再敲试验代码。

如果我们想改变这个   m,p的可写属性,可以在    lambda表达式中加入mutalbel关键字。

我画红线的部分,是原文书写错误,应该是有mutable时。

 这里强调一点,这个  const int n,在用这个mutable以后,那么对应的成员变量还是  const int n ,也就是说这个属性,是拿 mutable改变不了的。

3.lambda表达式中的  准函数,准对象,如何接收lambda表达式

当一个lambda表达式没有捕获任何外部变量时,则可以看成一个准函数,如果捕获了外部对象,则可以看成一个准对象。具体见下面图片。

 那么对于这种准对象的情况,我们是怎么接呢,c++  11中推出了std::function   模板类,它即能接准函数的情况,也能接准对象的情况。下面我们来简单的介绍下std::function.

std::function的实例可以对任何可以调用的目标实体进行存储、复制、和调用操作,这些目标实体包括普通函数、Lambda表达式、函数指针、以及其它函数对象等。std::function对象是对C++中现有的可调用实体的一种类型安全的包裹(我们知道像函数指针这类可调用实体,是类型不安全的)。

std::function包含于头文件 #include<functional> 中,可将各种可调用实体进行封装统一,包括

  • 普通函数
  • lambda表达式
  • 函数指针
  • 仿函数(functor 重载括号运算符实现)
  • 类成员函数
  • 静态成员函数

其中具体的使用方式,这里其他的就不再举例子,因为在我的另外一个文章中有介绍。这里就说一个lambda表达式的例子:

#include<memory>
#include<iostream>
#include<functional>
#include<map>
#include<string>
using namespace std;class SaveStuInfo{
public:void SaveInfo(const string& num,std::function<string(const string&,const string&)> callback){student_info.insert(make_pair(num,callback));
}void CallLmd(const string &num){string name = "zhc";string score = "100";student_info[num](name,score);
}
private:using get_name_score = std::function<string(const string &,const string &)>;std::map<std::string, get_name_score> student_info; //num as key};int main(void){
string num="123";
//string name ="zhc";
//string score="100";
auto lmd = [](const string&name,const string&score) ->string{cout<<"lmd was called value: "<<name+score<<endl;return name+score;
};
SaveStuInfo  temp;
temp.SaveInfo(num,lmd);
temp.CallLmd(num);
return 0;
}

这里要注意一点,在temp.SaveInfo(num,lmd),这个时候不要写成,temp.SaveInfo(num,lmd(name,score));这样会报错,换句话说,std::function,里不用存储具体的lambda参数里的值,真正传递的lambda参数是在调用这个lambda表达式的时候。

编译并且执行程序结果如下:

zhc@ubuntu:~/test/linux$ g++ -std=c++14  test.cpp -o test && ./test 
lmd was called value: zhc100


http://chatgpt.dhexx.cn/article/vxhVeTZx.shtml

相关文章

Java Lambda 表达式

目录 一、说明二、理解三、演示1.常规方法实现2.静态内部类3.局部内部类4.匿名内部类5.Lambda表达式6.Lambda再简化 一、说明 Lambda表达式是什么 Lambda 表达式也称为闭包&#xff0c;是Java 8 发布的新特性Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中) …

C++ Lambda表达式

在C11和更高版本中&#xff0c;Lambda表达式&#xff08;通常称为Lambda&#xff09;是一种在被调用的位置或作为参数传递给函数的位置定义匿名函数对象&#xff08;闭包&#xff09;的简便方法。Lambda通常用于封装传递给算法或异步函数的少量代码行。1 Lambda表达式是C11中一…

java Lambda表达式详解

文章目录 一、背景1.1语法1.2函数式接口 二、Lambda的基本使用2.1引子2.2常见的使用方式2.3语法小结 三、变量捕获3.1 匿名内部类3.2 匿名内部类的变量捕获3.3Lambda的变量捕获 四、Lambda在集合当中的使用4.1 Collection接口4.2 list接口4.3 Map接口 总结 提示&#xff1a;以下…

Lambda表达式详解

Lambda表达式 1. 为什么使用lambda表达式2. 入门案例3. lambda表达式组成4. lambda表达式使用4.1 语法格式一4.2 语法格式二4.3 语法格式三4.4 语法格式四4.5 语法格式五4.6 语法格式六 5. 总结 1. 为什么使用lambda表达式 lambda是一个匿名函数&#xff0c;我们可以吧lambda表…

Lambda 表达式

一.什么事Lambda表达式 Lambda 表达式是一种匿名函数&#xff0c;也可称为闭包&#xff0c;简单地说&#xff0c;它是没有声明的方法&#xff0c;也即没有访问修饰符、返回值声明和名字。 它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格&#xff0c;使 Java 语言…

Lambda表达式超详细总结(简单易懂)

文章目录 1、什么是Lambda表达式2、为什么使用Lambda表达式3、函数式接口&#xff08;lambda表达式的使用前提&#xff09;4、推导Lambda表达式5、Lambda表达式语法 1、什么是Lambda表达式 Lambda表达式&#xff0c;也可称为闭包。其本质属于函数式编程的概念&#xff0c;是Ja…

Lambda表达式超详细总结

文章目录 1. 什么是Lambda表达式2. 为什么使用Lambda表达式3. Lambda表达式语法4. 函数式接口4.1 什么是函数式接口4.2 自定义函数式接口4.3 Java内置函数式接口 5. 方法引用6. 构造器引用7. 数组引用8. Lambda表达式的作用域8.1 访问局部变量8.2 访问局部引用&#xff0c;静态…

SQL语句学习之SQL基础的表创建以及添加数据

SQL语句学习之SQL基础的表创建以及添加数据 学习目标1&#xff1a; 一周内掌握SQL基础语句 tip:主要是在牛客网&#xff08;牛客网&#xff09;上进行练习&#xff0c;里面有在线编程&#xff0c;可以直接运行&#xff0c;而且有解题的思路&#xff0c;比较清晰&#xff0c;而…

Hive SQL之表与建表

Hive数据模型总览 用户通过数据库访问Hive后&#xff0c;首先选择哪个数据库&#xff0c;然后在库的下面选择的是一张张表&#xff0c;表是管理数据的的最基本的所在&#xff0c;在表中的一行行记录&#xff0c;在现实中就是一条条数据&#xff0c;里面有我们的字段字段类型和它…

SQLite 创建表SQL语句

SQLite 创建表 创表语法 CREATE TABLE [表名称](--主键列不可为空[列1] [类型] PRIMARY KEY NOT NULL,--列可为空[列2] [类型],--列不可为空[列3] [类型] NOT NULL );创表示例 CREATE TABLE User (Id INT PRIMARY KEY NOT NULL,Name Text,Sex INT NOT NULL )在线Sqlite查看器…

SQL创建表为啥不显示

这里新建表 左上角报存 保存完以后点击刷新就会出来刚创建的表格

SQL 创建表的备份

1. SELECT INTO 语句 SELECT INTO 语句从一个表中选取数据&#xff0c;然后把数据插入另一个表中。 SELECT INTO 语句常用于创建表的备份复件或者用于对记录进行存档&#xff1b; SQL SELECT INTO 语法 您可以把所有的列插入新表&#xff1a; SELECT * INTO new_table_name…

SQL Server创建表

我们要怎么在数据库中创建表呢&#xff01;首先&#xff0c;表在数据库和模式中唯一命名。每个表包含一个或多个列。 每列都有一个相关的数据类型&#xff0c;用于定义它可以存储的数据类型&#xff0c;例如&#xff1a;数字&#xff0c;字符串和日期。 要创建新表&#xff0c;…

SQL Server 创建表

我们在上一节中完成了数据库的创建&#xff0c;在本节&#xff0c;我们要往这个新的数据库中加入点数据&#xff0c;要想将数据添加到数据库&#xff0c;我们就必须在数据库中添加一个表&#xff0c;接下来来看看具体的操作。 我们的数据库是一个任务跟踪数据库&#xff0c;那…

ORACLE SQL 创建表

1.创建表&#xff1a; 1.1表名和列名&#xff1a; 一定要以字母开头 一定在 1-30 个字符之间 只能包含 A–Z, a–z, 0–9, _, $, 和 # 一定不能和用户定义的其他对象重名 一定不能是Oracle 的保留字 一定要有CREATE TABLE权限 而且需要一定的存储空间 还要指定的&…

利用SQL创建表结构

一、创建图书管理系统&#xff0c;其中涉及到的对象有&#xff08;图书分类&#xff0c;图书&#xff0c;学生&#xff0c;借书记录&#xff09; 1、列出关系模式 (1) 书本类别&#xff08;种类编号&#xff0c;种类名称&#xff09; (2) 学生&#xff08;学生编号&#xff0…

SQL创建表

要创建新的表&#xff0c;就要使用create table语句。 1、第一&#xff0c;要指定数据库的名称&#xff0c;必须是数据库有的&#xff0c;如果没有指定&#xff0c;那就默认是当前数据库。&#xff08;如图下所示是没有指定数据库的&#xff09; 2、第二&#xff0c;指定表的模…

SQL表的创建

一&#xff0c;创建表 1.使用普通方法创建表 1&#xff0c;进入SQL进行连接 2&#xff0c;在左边会有一个对象资源管理器&#xff0c;右键数据库&#xff0c;在弹出的窗口中选择新建数据库 3&#xff0c;给这个包取个名字&#xff0c;在这个界面可以给这个表选择存储地方&…

SQL语句之表的创建和使用

表 一、表的创建&#xff08;DDL&#xff09;1.建表的语法格式创建一个学生表 2.mysql中的数据类型3.删除表 二、在表中插入数据insert&#xff08;DML&#xff09;1.insert2.insert插入日期3.date和datetime区别 三、修改(update)DML1.语法格式 四、删除数据(delete)DML1.语法…

SQL 创建数据库,创建表

1.SQL CREATE DATABASE 语法 CREATE DATABASE 库名;创建数据库后&#xff0c;您可以在数据库列表中检查它。 SHOW DATABASES;2.SQL CREATE TABLE 语句 CREATE TABLE 语句用于创建数据库中的表。表由行和列组成&#xff0c;每个表都必须有个表名。 SQL CREATE TABLE 语法 C…