当前位置: 首页 > news >正文

掌握现代 C++:Lambda 在 C++14、C++17 和 C++20 中的演变

一、背景

Lambda 是现代 C++ 最受欢迎的功能之一。自从在 C++ 11 中引入以来,它们在 C++ 代码中无处不在。而且,自从它们在 C++11 中出现以来,它们已经发展并获得了重要的功能。其中一些功能有助于编写更具表现力的代码,并且由于现在使用 lambda 非常普遍,因此花时间学习可以用它们做什么是非常值得的。

本文目标是剖析 lambda 的主要演变过程,但不是所有的小细节。对 lambda 的基础知识不了解可以阅读博主的另一篇文章,有详细介绍。

lambda 的演变一般是赋予它们手动定义函数对象的功能。

二、C++14 中的 Lambda

在 C++14 中,lambda 获得了 4 项主要增强功能:

  • 默认参数

  • 模板参数

  • 广义捕获

  • 从函数返回 lambda

2.1、默认参数

在 C++14 中,lambda 可以采用默认参数,就像任何函数一样:

代码语言:C++

自动换行

AI代码解释

auto myLambda = [](int x, int y = 0) { std::cout << x << '-' << y << '\n'; }; std::cout << myLambda(1, 2) << '\n'; std::cout << myLambda(1) << '\n';

输出:

代码语言:Bash

自动换行

AI代码解释

1-2 1-0

2.2、模板参数

在 C++11 中必须定义 lambda 参数的类型:

代码语言:C++

自动换行

AI代码解释

auto myLambda = [](int x){ std::cout << x << '\n'; };

在 C++14 中,可以让它们接受任何类型:

代码语言:C++

自动换行

AI代码解释

auto myLambda = [](auto&& x){ std::cout << x << '\n'; };

即使不需要处理多种类型,这对于避免重复并使代码更紧凑和可读也很有用。例如,这种 lambda:

代码语言:C++

自动换行

AI代码解释

auto myLambda = [](namespace1::namespace2::namespace3::ACertainTypeOfWidget const& widget) { std::cout << widget.value() << '\n'; };

变成:

代码语言:C++

自动换行

AI代码解释

auto myLambda = [](auto&& widget) { std::cout << widget.value() << '\n'; };

2.3、广义捕获

在 C++11 中,lambda 只能捕获其作用域中的现有对象:

代码语言:C++

自动换行

AI代码解释

int z = 42; auto myLambda = [z](int x){ std::cout << x << '-' << z + 2 << '\n'; };

C++14 借助强大的lambda广义捕获,可以用几乎任何东西初始化捕获的值。示例:

代码语言:C++

自动换行

AI代码解释

int z = 42; auto myLambda = [y = z + 2](int x) { std::cout << x << '-' << y << '\n'; }; myLambda(1);

输出:

代码语言:Bash

自动换行

AI代码解释

1-44

2.4、从函数返回 lambda

Lambda 受益于 C++14 的语言功能:从函数返回,而无需指定返回类型。由于 lambda 的类型是由编译器生成的,因此在 C++11 中无法从函数返回 lambda。

代码语言:C++

自动换行

AI代码解释

/* what type should we write here ?? */ f() { return [](int x){ return x * 2; }; }

在 C++14 中可以通过用作返回类型来返回 lambda。这在一段代码中间有一个大的 lambda 的情况下很有用。

展开

代码语言:C++

自动换行

AI代码解释

void f() { // ... int z = 42; auto myLambda = [z](int x) { // ... // ... // ... }; // ... }

可以将 lambda 打包到另一个函数中,从而引入另一个抽象级别:

展开

代码语言:C++

自动换行

AI代码解释

auto getMyLambda(int z) { return [z](int x) { // ... // ... // ... }; } void f() { // ... int z = 42; auto myLambda = getMyLambda(z); // ... }

三、C++17 中的 Lambda

C++17 为 lambda 带来了一个重大增强:它们可以声明constexpr

代码语言:C++

自动换行

AI代码解释

constexpr auto times2 = [] (int n) { return n * 2; };

然后,可以在编译时评估的上下文中使用此类 lambda:

代码语言:C++

自动换行

AI代码解释

static_assert(times2(3) == 6);

这在模板编程中特别有用。

注意:lambda 在 C++20 中变得更加有用。事实上,只有在 C++20 中,大多数 STL 算法才变得如此,并且它们可以与 lambda 一起使用,以创建在编译时评估的集合的复杂操作。不过,有一个例外:std::array非变异访问操作在 C++ 14 中立即变为std::array constexpr,而在 C++17 中变为非变异访问操作constexpr

lambda 在 C++17 中获得的另一个特性是捕获*this的副本的简单语法。示例:

代码语言:C++

自动换行

AI代码解释

struct MyType{ int m_value; auto getLambda() { return [this](){ return m_value; }; } };

此 lambda 捕获this指针的副本。如果 lambda 的生存期超过对象的生存期,则可能会导致内存错误,例如:

代码语言:C++

自动换行

AI代码解释

auto lambda = MyType{42}.getLambda(); lambda();

由于MyType在第一个语句的末尾被销毁,因此第二个语句调用的lambda取消了this引用访问其m_value,但this指向一个被销毁的对象。这会导致未定义的行为,通常是应用程序崩溃。

解决此问题是在lambda中捕获整个*this对象的副本。C++17 提供了语法来实现这一点。

展开

代码语言:C++

自动换行

AI代码解释

struct MyType { int m_value; auto getLambda() { return [*this](){ return m_value; }; } };

当然,在 C++ 14 中使用广义捕获已经可以实现相同的结果:

展开

代码语言:C++

自动换行

AI代码解释

struct MyType { int m_value; auto getLambda() { return [self = *this](){ return self.m_value; }; } };

只是C++17 使语法更好。

四、C++20 中的 Lambda

Lambda 在 C++ 20 中得到进一步发展,但其功能不如 C++ 或 C++ 17 那么基本。C++ 20 中 lambda 的一个增强功能是定义模板的经典语法,使它们更接近手动定义的函数对象:

代码语言:C++

自动换行

AI代码解释

auto myLambda = []<typename T>(T&& value){ std::cout << value << '\n'; };

这使得访问模板参数类型比使用 表达式(如auto&&)的 C++ lambda 模板更容易。

另一个改进是能够捕获可变参数包:

代码语言:C++

自动换行

AI代码解释

template<typename... Ts> void f(Ts&&... args) { auto myLambda = [...args = std::forward<Ts>(args)](){}; }

五、总结

lambda从C++14到C++20都有了不少的改进。但也还有更多没有总结进来。这些主要功能伴随着许多小特性让 lambda 代码更容易编写。深入研究 lambda 是更好了解 C++ 语言,值得投入时间。


http://www.jsqmd.com/news/752556/

相关文章:

  • 革命性岛屿设计工具:Happy Island Designer深度解析与进阶应用
  • 终极CPUDoc性能优化指南:免费解锁CPU隐藏性能的完整教程
  • 三步搞定城通网盘下载:免费高效的直连解析终极方案
  • 如何3分钟快速部署个人视频下载神器:VideoDownloadHelper完整指南
  • 别再死记硬背数码管段码了!用STC89C52+S8550三极管,从原理到代码彻底搞懂共阳/共阴驱动
  • 2026届必备的六大降重复率平台实测分析
  • Docker Compose 安装 Etcd
  • 微信小程序虚拟支付全解:规则、接入与合规
  • 手把手教你用pyinstxtractor和uncompyle6找回丢失的Python源码(附Python 3.8及以下版本完整流程)
  • ArcGIS 10.8安装后必做的5项设置与优化,让你的软件运行更流畅
  • US Cities Are Axing Flock Safety Surveillance Technology: 当监控之眼被蒙上,我们在守护什么?
  • 【微软内部PPT首次流出】.NET 9 Configuration 3.0架构图解:低代码≠无代码,而是编译期验证+运行时热重载
  • 闲鱼数据采集:基于UI自动化的逆向工程实践
  • 2026届毕业生推荐的十大降AI率神器推荐榜单
  • 如何将PowerPoint演示文稿一键转换为现代网页?PPTX2HTML解密
  • 观察在虚拟机环境下使用Taotoken调用大模型的延迟与稳定性表现
  • 抖音视频怎么在线去水印?抖音视频在线去水印方法实测+2026最新 在线去水印工具推荐 - 爱上科技热点
  • 观察通过Taotoken调用不同模型时的token消耗与成本明细
  • ThinkPHP 模板引擎编译缓存如何清理避免页面显示旧数据?
  • 2025届最火的六大降重复率网站推荐榜单
  • 嵌入式安全必修课:搞懂SRAM的ECC,别让你的车规MCU在关键时刻‘掉链子’
  • 免费的小红书去水印工具效果最好?2026最新年强烈推荐 - 爱上科技热点
  • AppleRa1n:解锁iOS设备激活锁的实用指南
  • LeetCode 18.四数之和
  • 最新媒体新闻稿发稿平台有哪些?如何选择最适合的发布渠道? - 代码非世界
  • 长期使用中感受到的 Taotoken 服务稳定性与开发者支持
  • WarcraftHelper终极指南:三步快速提升魔兽争霸III游戏体验
  • ARM调试寄存器OSLSR与OSSRR深度解析
  • Sunshine游戏串流:5步搭建你的个人云游戏服务器终极指南
  • 视频去水印免费工具怎么选?2026最新实测优缺点对比,视频去水印免费推荐 - 爱上科技热点