C++匿名函数

C++ 中的匿名函数(Lambda 表达式)是 C++11 引入的一项重要特性,它允许你在需要的地方定义一个临时的、无名的函数对象,使代码更加简洁和灵活。

1. 基本语法

Lambda 表达式的基本结构:

[capture list](parameter list) -> return type { function body }
  • [capture list]:捕获外部变量,指定如何将外部作用域的变量传递给 lambda。
  • (parameter list):参数列表,与普通函数的参数类似(可省略,但若省略括号必须为空)。
  • -> return type:返回类型(可省略,编译器会自动推导)。
  • { function body }:函数体,包含具体的实现逻辑。

示例

auto add = [](int a, int b) -> int { return a + b; };
int result = add(3, 4);  // 结果为 7

2. 捕获列表(Capture List)

捕获列表用于访问外部作用域中的变量,有以下几种方式:

值捕获(By Value)
  • 使用 [var] 捕获变量的副本。
  • Lambda 创建时拷贝变量,后续修改不影响 lambda 内部的值。
int x = 10;
auto lambda = [x]() { return x * 2; };  // 捕获 x 的值
x = 20;
std::cout << lambda();  // 输出 20(捕获的是 x 的副本)
引用捕获(By Reference)
  • 使用 [&var] 捕获变量的引用。
  • Lambda 内部使用的是变量的引用,外部修改会影响 lambda 内部。
int x = 10;
auto lambda = [&x]() { return x * 2; };  // 捕获 x 的引用
x = 20;
std::cout << lambda();  // 输出 40(引用 x 的当前值)
隐式捕获
  • 使用 [=] 捕获所有外部变量的值(值捕获)。
  • 使用 [&] 捕获所有外部变量的引用(引用捕获)。
int a = 5, b = 10;
auto lambda = [=]() { return a + b; };  // 值捕获 a 和 b
auto lambda2 = [&]() { a++; return a + b; };  // 引用捕获 a 和 b
混合捕获
  • 同时使用值捕获和引用捕获,例如 [=, &a](默认值捕获,a 引用捕获)。
int a = 5, b = 10;
auto lambda = [=, &a]() { a++; return a + b; };  // a 引用捕获,b 值捕获

3. 参数列表

Lambda 的参数列表与普通函数类似,但不支持默认参数。

auto greet = [](const std::string& name) {std::cout << "Hello, " << name << "!" << std::endl;
};
greet("Alice");  // 输出 "Hello, Alice!"

4. 返回类型

返回类型可省略,编译器会自动推导。若需要显式指定,使用 -> type

auto sum = [](int a, int b) -> int { return a + b; };  // 显式指定返回类型
auto square = [](double x) { return x * x; };  // 自动推导返回类型

5. 可变 Lambda(Mutable Lambda)

默认情况下,值捕获的变量在 lambda 内部是只读的。使用 mutable 关键字可修改值捕获的变量。

int x = 10;
auto lambda = [x]() mutable {x++;  // 允许修改值捕获的 xreturn x;
};
std::cout << lambda();  // 输出 11(但外部 x 仍为 10)

6. 泛型 Lambda(C++14+)

使用 auto 作为参数类型,使 lambda 成为泛型函数。

auto print = [](const auto& value) {std::cout << value << std::endl;
};
print(42);      // 输出整数
print("test");  // 输出字符串

7. 捕获 this 指针

在类成员函数中,可捕获 this 指针以访问类的成员变量和方法。

class MyClass {
public:int value = 10;void func() {auto lambda = [this]() { return value * 2; };std::cout << lambda();  // 输出 20}
};

8. 捕获初始化(C++14+)

允许在捕获列表中初始化新变量,可移动构造对象或重命名捕获的变量。

int x = 10;
auto lambda = [y = x + 5]() { return y; };  // 初始化 y 为 15
std::cout << lambda();  // 输出 15// 移动捕获(适用于不可复制的对象,如 std::unique_ptr)
auto ptr = std::make_unique<int>(42);
auto lambda2 = [ptr = std::move(ptr)]() { return *ptr; };

9. Lambda 的类型和存储

  • Lambda 表达式的类型是一个唯一的、未命名的闭包类型(Closure Type)。
  • 可使用 autostd::function 存储 lambda。
// 使用 auto(推荐,效率更高)
auto add = [](int a, int b) { return a + b; };// 使用 std::function(需包含 <functional>)
std::function<int(int, int)> multiply = [](int a, int b) { return a * b; };

10. Lambda 在 STL 中的应用

Lambda 常用于简化 STL 算法的使用。

#include <algorithm>
#include <vector>std::vector<int> nums = {1, 2, 3, 4, 5};// 使用 lambda 作为谓词
auto sum = std::accumulate(nums.begin(), nums.end(), 0, [](int acc, int x) { return acc + x; });// 排序
std::sort(nums.begin(), nums.end(), [](int a, int b) { return a > b; });  // 降序排序// 查找第一个大于 3 的元素
auto it = std::find_if(nums.begin(), nums.end(), [](int x) { return x > 3; });

11. 常量表达式 Lambda(C++17+)

使用 constexpr 关键字使 lambda 可以在编译时求值。

constexpr auto add = [](int a, int b) { return a + b; };
static_assert(add(3, 4) == 7, "Error");  // 编译时检查

12. 模板 Lambda(C++20+)

使用模板参数(template <typename T> 的简写)使 lambda 更灵活。

auto lambda = []<typename T>(const T& a, const T& b) { return a + b; };
int sum_int = lambda(1, 2);        // T 推导为 int
double sum_double = lambda(1.5, 2.5);  // T 推导为 double

13. 异常规范(C++17 前)

使用 noexcept 指定 lambda 是否抛出异常。

auto safe_divide = [](double a, double b) noexcept {return b != 0 ? a / b : 0;
};

14. 性能考虑

  • Lambda 通常比普通函数指针或 std::function 更高效,因为编译器可内联其代码。
  • 值捕获会复制变量,可能影响性能(尤其是大对象),此时应优先使用引用捕获。

总结

C++ 的匿名函数(Lambda)提供了强大而灵活的语法,使代码更简洁、更易读。掌握捕获列表、参数、返回类型和各种特性(如泛型、捕获初始化)是使用 Lambda 的关键。合理使用 Lambda 可以显著提升 C++ 代码的表达力和效率。

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

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

相关文章

LabVIEW机械振动信号分析与故障诊断

利用 LabVIEW 开发机械振动信号分析与故障诊断系统&#xff0c;融合小波变换、时频分布、高阶统计量&#xff08;双谱&#xff09;等先进信号处理技术&#xff0c;实现对齿轮、发动机等机械部件的非平稳非高斯振动信号的特征提取与故障诊断。系统通过虚拟仪器技术将理论算法转化…

湖北理元理律师事务所:债务优化如何实现“减负不降质”?

在债务压力普遍加剧的背景下&#xff0c;如何平衡债务清偿与生活质量&#xff0c;成为个人及企业关注的焦点。湖北理元理律师事务所基于多年实务经验&#xff0c;总结出一套“法律财务”双轨制债务优化模型&#xff0c;其核心在于通过科学规划&#xff0c;帮助债务人在法律框架…

多链互操作性标准解析:构建下一代区块链互联生态

引言 在区块链技术快速演进的今天&#xff0c;“多链宇宙”已成为不可逆的趋势。然而&#xff0c;链与链之间的孤立性导致流动性割裂、开发成本高昂和用户体验碎片化。互操作性标准的制定&#xff0c;正是打破这一僵局的核心钥匙。本文将深入探讨主流互操作性协议的技术架构、…

电脑开机提示按f1原因分析及解决方法(6种解决方法)

经常有网友问到一个问题,我电脑开机后提示按f1怎么解决?不管理是台式电脑,还是笔记本,都有可能会遇到开机需要按F1,才能进入系统的问题,引起这个问题的原因比较多,今天小编在这里给大家列举了比较常见的几种电脑开机提示按f1的解决方法。 电脑开机提示按f1原因分析及解决…

讲讲git 和svn

讲讲git 和svn 目录Git到底是什么&#xff1f;它该怎末用&#xff1f;核心概念基础操作1. 仓库的创建2. 文件的提交工作流程3. 分支管理4. 远程仓库操作 进阶操作实际应用建议**基本用法****常用命令的帮助示例****帮助文档的结构****替代方法****练习建议****核心概念****与Gi…

【行为型之中介者模式】游戏开发实战——Unity复杂系统协调与通信架构的核心秘诀

文章目录 &#x1f54a;️ 中介者模式&#xff08;Mediator Pattern&#xff09;深度解析一、模式本质与核心价值二、经典UML结构三、Unity实战代码&#xff08;成就系统协调&#xff09;1. 定义中介者接口与同事基类2. 实现具体同事类3. 实现具体中介者4. 客户端使用 四、模式…

【网工第6版】第10章 网络规划和设计②

目录 ■ 网络分析与设计 ◆ 网络规划设计模型 ◆ 网络设计的约束因素 ◆ 技术评价 ■ 网络结构与功能 ◆ 局域网结构类型 ■ 网络冗余设计 ◆ 网络冗余设计-备用路径 ◆ 网络冗余设计-负载分担 ■ 广域网接入技术 ◆ 广域网接入技术 ◆ 接入和终结设备 ■ 网络分析…

PowerBI链接EXCEL实现自动化报表

PowerBI链接EXCEL实现自动化报表 曾经我将工作中一天的工作缩短至2个小时&#xff0c;其中最关键的一步就是使用PowerBI链接Excel做成一个自动化报表&#xff0c;PowerBI更新源数据&#xff0c;Excel更新报表并且保留报表格式。 以制作一个超市销售报表为例&#xff0c;简单叙…

C#.NET 或 VB.NET Windows 窗体中的 DataGridView – 技巧、窍门和常见问题

DataGridView 控件是一个 Windows 窗体控件&#xff0c;它允许您自定义和编辑表格数据。它提供了许多属性、方法和事件来自定义其外观和行为。在本文中&#xff0c;我们将讨论一些常见问题及其解决方案。这些问题来自各种来源&#xff0c;包括一些新闻组、MSDN 网站以及一些由我…

表记录的检索

1.select语句的语法格式 select 字段列表 from 表名 where 条件表达式 group by 分组字段 [having 条件表达式] order by 排序字段 [asc|desc];说明&#xff1a; from 子句用于指定检索的数据源 where子句用于指定记录的过滤条件 group by 子句用于对检索的数据进行分组 ha…

能源设备数据采集

在全球可持续发展目标与环境保护理念日益深入人心的时代背景下&#xff0c;有效管理和优化能源使用已成为企业实现绿色转型、提升竞争力的关键路径。能源设备数据采集系统&#xff0c;作为能源管理的核心技术支撑&#xff0c;通过对各类能源生产设备运行数据的全面收集、深度分…

【鸿蒙开发】性能优化

语言层面的优化 使用明确的数据类型&#xff0c;避免使用模糊的数据类型&#xff0c;例如ESObject。 使用AOT模式 AOT就是提前编译&#xff0c;将字节码提前编译成机器码&#xff0c;这样可以充分优化&#xff0c;从而加快执行速度。 未启用AOT时&#xff0c;一边运行一边进…

群晖NAS部署PlaylistDL音乐下载器结合cpolar搭建私有云音乐库

文章目录 前言1.关于PlaylistDL音乐下载器2.Docker部署3.PlaylistDL简单使用4.群晖安装Cpolar工具5.创建PlaylistDL音乐下载器的公网地址6.配置固定公网地址总结 前言 各位小伙伴们&#xff0c;你们是不是经常为了听几首歌而开通各种平台的VIP&#xff1f;或者为了下载无损音质…

REST架构风格介绍

一.REST&#xff08;表述性状态转移&#xff09; 1.定义 REST&#xff08;Representational State Transfer&#xff09;是由 Roy Fielding 在 2000 年提出的一种软件架构风格&#xff0c;用于设计网络应用的通信模式。它基于 HTTP 协议&#xff0c;强调通过统一的接口&#…

计算机视觉----基于锚点的车道线检测、从Line-CNN到CLRNet到CLRKDNet 本文所提算法Line-CNN 后续会更新以下全部算法

本文所提算法如下&#xff1a; 叙述按时间顺序 你也可以把本文当作快速阅读这几篇文献的一个途径 所有重要的部分我都已经标注并弄懂其原理 方便自己也是方便大家 Line-CNN&#xff1a;基于线提议单元的端到端交通线检测 摘要 交通线检测是一项基础且具有挑战性的任务。以往的…

一.android Studio开发系统应用——导入TvSettings源码

目标 最终效果如上,实现在AS中编辑源码后一键在真机中运行。达到和普通应用开发一样的调试和编码过程。这种方法可以大幅度提升开发速度,但是导入过程确实相对繁琐和消耗时间。适合需要精细或者频繁改动的系统app源码。 一、新建项目 包名:com.android.tv.settings 版本:…

20250515让飞凌的OK3588-C的核心板在Linux R4下适配以太网RTL8211F-CG为4线百兆时的接线图

20250515让飞凌的OK3588-C的核心板在Linux R4下适配以太网RTL8211F-CG为4线百兆时的接线图 2025/5/15 20:19 缘起&#xff1a;以前做的网线找不到了&#xff0c;那就再来一条吧。 引脚定义要从头来过&#xff1f;还好找到了一条。 开干&#xff01; 万用表一对/点&#xff0c;几…

【技术原理】Linux 文件时间属性详解:Access、Modify、Change 的区别与联系

在 Linux 系统中&#xff0c;每个文件都有三个核心时间属性&#xff1a;Access Time (atime)、Modify Time (mtime) 和 Change Time (ctime)。它们分别记录文件不同维度的变更信息&#xff0c;以下是具体区别与联系&#xff1a; 一、定义与触发条件 时间属性定义触发条件示例A…

乘法口诀练习神器

请你利用python语言开发一个“乘法口诀练习神器”&#xff0c;主要辅助小学生练习乘法口诀&#xff0c;主要功能如下&#xff1a; 1. 能够随机循环出10道题&#xff0c;可以是乘法或者是除法。如果是乘法&#xff0c;确保两个因数都是1-9之间的整数&#xff1b;如果是除法&…

[c语言日寄]数据结构:栈

【作者主页】siy2333 【专栏介绍】⌈c语言日寄⌋&#xff1a;这是一个专注于C语言刷题的专栏&#xff0c;精选题目&#xff0c;搭配详细题解、拓展算法。从基础语法到复杂算法&#xff0c;题目涉及的知识点全面覆盖&#xff0c;助力你系统提升。无论你是初学者&#xff0c;还是…