C++11中的lambda匿名函数

news2024/9/27 12:15:21

一、引言

        C++11中引入的lambda表达式(也称为匿名函数或lambda函数)提供了一种方便且灵活的方式来定义和使用小的匿名函数对象。这些lambda表达式在编写回调函数、操作容器的算法中、或者在需要快速定义和传递一个函数逻辑给另一个函数时特别有用。

二、基本语法

[capture](parameters) mutable -> return_type {  
    // 函数体  
}
  • capture:对lambda函数外部变量的捕获方式:
lambda匿名函数捕获外部变量的方式
表示方式说明
[ ]不捕获任何外部变量
[=]值传递的方式捕获全部外部变量
[&]引用传递的方式捕获全部外部变量
[&x, y]x以引用传递的方式捕获,y以传值的方式捕获
  • parameters:函数的参数列表
  • mutable:是否可以修改以值传递的外部变量,加上mutable表示可以修改,否则不能修改
  • return_type:函数的返回类型,省略时,返回类型就是函数内return语句返回值的类型。

说明:全局变量会自动进行引用捕获。

三、外部变量的捕获

1. 不捕获任何外部变量

int main(){
	
	auto fun = [](int a, int b)->int{
		return a + b;
	};
	
	int sum = fun(1, 2);
	
	cout << sum << endl;  // 输出3
	
	return 0;
}

2. 访问全局变量

int x = 10;    // 全局变量

int main(){	
	auto fun = [](){
		++x;    // 自动捕获全局变量 x
	};
	
	cout << x << endl;    // 输出10
	
	fun();
	
	cout << x << endl;    // 输出11
	
	return 0;
} 

3. 传值捕获所有外部变量

int main(){	
	int x = 10;
	int y = 20;
	auto fun = [=]()->int{
		// x++;  //不可以修改x的值,x在lambda内是const的 
		return x + y;    // x得到30
	};
	
	int sum = fun();
	cout << sum << endl;
	
	return 0;
} 

输出内容: 

         从输出的地址可以知道,lambda函数内的变量x与函数外的变量x不是同一个变量,只是用外部x变量的值初始化了函数内的x变量(相当于传值的形参)。并且不能修改x的值,因为x是const的。如果想要修改x的值,可以使用mutable关键字修饰lambda函数。

4. 以引用的方式捕获所有的外部变量

int main(){	
	int x = 10;
	int y = 20;
	cout << (&x) << endl;
	auto fun = [&]()->int{
		x++;  //以引用方式捕获的变量可以修改其值
		cout << (&x) << endl; 
		return x + y;
	};
	
	int sum = fun();
	cout << sum << endl;
	
	return 0;
}

输出内容:

        从输出的地址可以知道,lambda函数内的变量x与函数外的变量x是同一个变量(地址相同)。 

5. 混合捕获

int main(){	
	int x = 10;
	int y = 20;
	cout << (&x) << endl;
	auto fun = [&x, y]()mutable{
		x++;  // 可以修改x的值,修改的是函数外的x变量 
		y++;  // y以传值捕获,因为函数声明为mutable,所有y可以修改,修改的是lambda内部的y 
		cout << (&x) << endl; 
		return x + y;
	};
	
	int sum = fun();
	cout << sum << endl;
	cout << "x = " << x << ", y = " << y << endl;
	
	return 0;
} 

输出内容:

         x以引用传递,y以值传递,所以在lambda函数内修改的x是函数外的x,而y则是lambda函数内的y,与函数外的y不是同一个变量。

四、参数示例

        lambda函数的参数和普通函数是一样的,可以有传值、传地址和传引用几种方式。

1. 传值

int main(){	
	int x = 10;
	int y = 20;
	cout << (&x) << endl;
	auto fun = [](int x, int y){
		x++;  // 可以修改形参x的值
		y++;  // 可以修改形参y的值 
		cout << (&x) << endl; 
		return x + y;
	};
	
	int sum = fun(x, y);
	cout << sum << endl;
	// lambda函数外的x和y没有被修改 
	cout << "x = " << x << ", y = " << y << endl;
	
	return 0;
} 

2. 传地址

int main(){	
	int x = 10;
	int y = 20;
	auto fun = [](int *x, int *y){
		(*x)++;  
		(*y)++;  
		return (*x) + (*y);
	};
	
	int sum = fun(&x, &y);
	cout << sum << endl;
	// x = 11, y = 21 
	cout << "x = " << x << ", y = " << y << endl;
	
	return 0;
} 

3. 传引用

int main(){	
	int x = 10;
	int y = 20;
	auto fun = [](int &x, int &y){
		x++;  
		y++;  
		return x + y;
	};
	
	int sum = fun(x, y);
	cout << sum << endl;
	// x = 11, y = 21 
	cout << "x = " << x << ", y = " << y << endl;
	
	return 0;
} 

六、常用场合

1. 临时函数

// lambda表达式
#include <iostream>
#include <cstdlib>
using namespace std;

int main() {
	int arr[]{4,1,8,5,2};
	qsort(arr, sizeof(arr)/sizeof(arr[0]), sizeof(arr[0]), 
		[](const void *p1, const void *p2){
			int a = *((int*)p1);
			int b = *((int*)p2);
			return a == b ? 0 : (a < b ? 1 : -1);
		});
	
	for (int i: arr){
		cout << i << "  ";
	}  // 降序排列,输出8 5 4 2 1

	return 0;
}

2. STL算法中

#include <iostream>  
#include <vector>  
#include <algorithm>  
  
int main() {  
    std::vector<int> numbers = {1, 2, 3, 4, 5};  
  
    // 使用lambda表达式对vector中的每个元素加1  
    std::for_each(numbers.begin(), numbers.end(), [](int& n) { n += 1; });  
  
    // 打印结果  
    for(int n : numbers) {  
        std::cout << n << ' ';  
    }  
    std::cout << '\n';  
  
    return 0;  
}

七、注意事项


        Lambda表达式产生的对象类型是唯一的匿名非联合非聚合类类型,称为闭包类型。
捕获的变量(如果是引用捕获)在lambda函数体之外的生命周期结束时,引用可能变为悬空引用,使用时需要特别注意。
Lambda表达式中的this指针在捕获列表中自动处理,如果使用[&]或[=]捕获方式,则this指针会被捕获。如果要显式捕获this,可以在捕获列表中包含*this或this(视具体需求而定)。

八、小结


        C++11中的lambda表达式是C++编程中一个非常强大的特性,极大地增强了代码的灵活性和表达能力。同时增强了函数式编程的能力。

附:c++11新增的其他性

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2089384.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

行得稳,跑得远,美团如何做到长期主义持续发力?

在今天这样一个充满不确定性的时代&#xff0c;最能确定的是什么呢&#xff1f;我们这一代人&#xff0c;有幸成为诸多历史性时刻的亲历者和见证者。在此背景下&#xff0c;越来越多的企业家和经营者开始关注回归经营的本质和实现稳健经营的问题。 企业的稳健经营贯穿于企业的…

联华证券-掌握尾盘买入法的多种策略与实用技巧

尾盘买入法的常见方法与具体技巧 尾盘买入法是指在股票交易日的尾盘时段&#xff08;通常指收盘前的15-30分钟&#xff09;进行买入操作的策略。由于尾盘时段能够反映市场全天的交易情况以及主力资金的意图&#xff0c;尾盘买入法在捕捉短线机会方面具有一定的优势。以下是尾盘…

Python编码—掌握Python与Kubernetes:构建高效微服务架构

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

涨粉20w!AI真人视频转动漫,太好涨粉了,3步学会AI视频转绘!

大家好&#xff0c;我是程序员X小鹿&#xff0c;前互联网大厂程序员&#xff0c;自由职业2年&#xff0c;也一名 AIGC 爱好者&#xff0c;持续分享更多前沿的「AI 工具」和「AI副业玩法」&#xff0c;欢迎一起交流~ 这类视频&#xff0c;流量超大。从去年一直火到了现在。 前有…

dubbo:巧用dubbo的SPI机制实现各类定制功能(六)

文章目录 0. 引言1. dubbo SPI机制1.1 什么是SPI1.2 java的SPI机制1.3 dubbo的SPI机制1.4 dubbo的SPI使用 2. 基于SPI实现定制功能2.1 dubbo的上下文参数2.2 实现权限校验、参数日志等功能2.3 dubbo spi条件激活机制Activate 3. dubbo的SPI与java的SPI有什么区别4. 总结 0. 引言…

机器学习:词向量转换及代码实现

CountVectorizer 是 scikit-learn 库中的一个文本向量化工具&#xff0c;它将文本数据转换为词频特征矩阵。以下是 CountVectorizer 的算法原理和步骤&#xff1a; 原理 文本预处理&#xff1a; 分词&#xff1a;将文本分割成单词或短语&#xff08;tokens&#xff09;。小写化…

mysql配置与日常使用

mysql配置 1.从官网下载zip文件并解压&#xff0c;解压的路径中最好不要有中文 2.添加环境变量 这一步的作用是可以在任意文件夹下操作mysql指令 操作步骤&#xff1a;环境变量 ->用户变量Path -> 添加mysql的bin目录 V 3. 在mysql解压路径下添加配置文件my.ini …

力扣刷题第二阶段Days35(javascript)

1&#xff0c;题目描述 -- 和为K的子数组 给你一个整数数组 nums 和一个整数 k &#xff0c;请你统计并返回 该数组中和为 k 的子数组的个数 。 子数组是数组中元素的连续非空序列。 2,代码详解 哈希&#xff0b;前缀和 /*** param {number[]} nums* param {number} k* ret…

【openwrt-21.02】T750 openwrt-21.02 Linux-5.4.238 input子系统----gpio-keys实现分析

input子系统 输入子系统是由设备驱动层(input driver)、输入核心层(input core)、输入事件处理层(input event handle)组成 input子系统架构图 gpio-keys gpio-keys是基于input子系统实现的一个通用按键驱动,该驱动也符合linux驱动实现模型,即driver和device分离模型.一…

MIT 6.5840(6.824) Lab 4:Fault-tolerant Key/Value Service 设计实现

1 实验要求 本实验旨在利用lab 3中的Raft库&#xff0c;构建一个具备容错能力的键值存储服务。服务将作为一个复制状态机&#xff0c;由多个服务器组成&#xff0c;各服务器通过Raft协议同步数据库状态。即使在部分故障或网络隔离的情况下&#xff0c;只要大多数服务器正常&am…

通过MindOpt APL建模求解组合优化问题中的常见问题:图着色问题

组合优化问题&#xff1a;图着色问题 通过MindOpt APL建模求解组合优化问题中的常见问题&#xff1a;图着色问题 1. 背景知识 1.1. 组合优化问题 在之前发布的《组合优化问题&#xff1a;装箱问题》中&#xff0c;我们讲解了什么是组合优化&#xff08;Combinatorial Optimi…

【练习1】数字统计

题目&#xff1a; 分析&#xff1a; 枚举法数字拆分&#xff08;tmp % 10&#xff0c;tmp / 10&#xff09; 代码实现&#xff1a; public class Main {public static void main(String[] args) {Scanner in new Scanner(System.in);int l in.nextInt(), r in.nextInt();…

【2024 CCF编程能力等级认证(GESP)Python 】一级大纲

目录 1. 背景2. 考核知识块3. 考核内容3.1 计算机基础知识3.2 编程规范3.3 基础语法3.4 数据类型3.5 三大基本结构3.6 运算符3.7 模块导入与输入输出3.8 Turtle绘图4. 考核目标5. 题型分布6. 考试时长7. 认证时间与报名8. 政策与福利9. GESP一级认证形式 1. 背景 官网&#xff…

11 对话模型微调

提问&#xff1a;其实我一直觉的数据是最费事的一个&#xff0c;现在都是使用别人的数据&#xff0c;如果对于实际场景中那么我们该如何获取处理数据呢&#xff01; 1 数据处理&#xff1b; 2 模型选择&#xff0c;调参数&#xff1b; 数据 llm-wizard/alpaca-gpt4-data-zh …

简单的二叉树问题——二叉树的最大深度

给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;3示例 2&#xff1a; 输入&#xff1a;root [1,null,2] 输出…

【二叉树进阶】--- 前中后序遍历非递归

Welcome to 9ilks Code World (๑•́ ₃ •̀๑) 个人主页: 9ilk (๑•́ ₃ •̀๑) 文章专栏&#xff1a; 算法Journey 本篇博客我们将来了解有关二叉树前中后序遍历的非递归版本。 &#x1f3e0; 前序遍历 要迭代非递归实现二叉树的前序遍历&#xff0c;首先还…

【pgAdmin4】创建/删除:数据库Database和数据库表Table

目录 0.环境 1.简介 2.详细步骤 1&#xff09;创建数据库 法一&#xff1a;UI界面创建 法二&#xff1a;sql语句创建数据库 2&#xff09;创建数据库表 查看数据库表 查看数据库表内容 法一&#xff1a;UI界面创建数据库表 法二&#xff1a;sql语句创建数据库表 3&…

C_09_字符操作

字符串相关函数 头文件都是 string.h 概述&#xff1a; 由系统提供的由字符串处理的函数 属于库函数 所属头文件 string.h 1 strlen 测量字符串长度 语法&#xff1a; size_t strlen(const char *s);参数&#xff1a;要测量的字符串返回值&#xff1a;长度注意:测量的长度…

Screenshot Software,截屏软件

一.截屏软件 1.1 自带的一些截屏软件 &#xff08;1&#xff09;微信 &#xff08;2&#xff09;QQ 有一次无意中测试了下&#xff0c;截屏软件的截屏质量&#xff0c;发现对于同一个页面截全屏&#xff0c;微信截的屏质量相对于 win 自带的截图软件还要好的&#xff0c;所以…

Elasticsearch 开放推理 API 增加了对 Anthropic 的 Claude 的支持

作者&#xff1a;来自 Elastic Jonathan Buttner 我们很高兴地宣布 Elasticsearch Open Inference API 的最新功能&#xff1a;集成 Anthropic 的 Claude。这项功能使 Elastic 用户能够直接连接到 Anthropic 平台&#xff0c;并使用 Claude 3.5 Sonnet 等大型语言模型来构建 Ge…