C++23 让 Lambda 表达式中的 () 更可选:P1102R2 提案深度解析

news2025/4/22 9:32:35

文章目录

    • 一、背景与动机:Lambda 表达式中的痛点
      • 1.1 问题的根源
    • 二、P1102R2 提案:让 `()` 可选
      • 2.1 提案的核心内容
      • 2.2 语法调整的细节
      • 2.3 提案的合理性
    • 三、编译器支持:主流编译器的跟进
    • 四、对 C++ 编程的影响:简化语法与提升一致性
      • 4.1 简化语法
      • 4.2 提升语言一致性
      • 4.3 与其他新特性的结合
      • 4.4 示例代码
    • 五、总结:更简洁、更一致的 Lambda 表达式

在 C++23 标准中,对 Lambda 表达式的一项重大改进是让 () 在更多情况下可选。这一改进主要得益于 P1102R2 提案(Down with ()!)。本文将深入探讨这一变化的背景、实现细节、对编程实践的影响,以及主流编译器的支持情况。

一、背景与动机:Lambda 表达式中的痛点

Lambda 表达式是 C++11 引入的一种强大的匿名函数功能,它极大地简化了函数对象的使用场景。然而,在 C++20 及之前的版本中,Lambda 表达式的语法存在一些限制,尤其是在 () 的使用上。

在 C++20 中,Lambda 表达式在某些情况下必须显式写出空的 (),即使它没有参数。具体来说,当 Lambda 表达式包含以下内容时,() 不能省略:

  • mutable 修饰符
  • constexprconsteval 修饰符
  • 模板参数
  • 异常说明(如 noexcept
  • 属性(如 [[nodiscard]]
  • 尾返回类型
  • requires 子句

例如,以下代码在 C++20 中是非法的:

auto l = [] mutable {};

这是因为 mutable 修饰符要求必须显式写出空的 ()。这种规则不仅增加了学习和使用的复杂性,还容易导致开发者在编写代码时出现语法错误。

1.1 问题的根源

这种限制的根源在于 C++20 的语法设计。在 C++20 中,Lambda 表达式的语法定义为:

lambda-expression:
    lambda-introducer lambda-declarator[opt] compound-statement

其中,lambda-declarator 是可选的,但当它存在时,必须显式写出空的 ()。这种设计导致了不一致的语法规则,使得开发者需要记住哪些情况下可以省略 (),哪些情况下不能。

二、P1102R2 提案:让 () 可选

为了解决这一问题,P1102R2 提案(Down with ()!)被提出。该提案的目标是让 Lambda 表达式中的 () 在更多情况下可选,从而使语言更加一致和简洁。

2.1 提案的核心内容

P1102R2 提案的核心内容是调整 Lambda 表达式的语法定义,允许在更多场景下省略空的 ()。具体来说,无论 Lambda 表达式是否包含模板参数、修饰符、属性、尾返回类型或 requires 子句,都可以选择不写空的 ()

例如,以下代码在 C++23 中都是合法的:

auto l1 = [] mutable {};
auto l2 = [] constexpr {};
auto l3 = [] noexcept {};
auto l4 = [] [[nodiscard]] {};
auto l5 = [] -> int { return 42; };

2.2 语法调整的细节

为了实现这一目标,提案对 Lambda 表达式的语法定义进行了修改。具体调整如下:

  1. 移除 lambda-declaratoropt 标记:在 C++20 中,lambda-declarator 是可选的,但当它存在时,必须显式写出空的 ()。在 C++23 中,移除了 opt 标记,因为现在 lambda-declarator 可以为空,该标记变得多余。
  2. 明确所有 Lambda 表达式都有一个 lambda-declarator:即使它为空,这也使得语法定义更加清晰和一致。

2.3 提案的合理性

提案还详细讨论了这一改变的合理性。例如,它指出,即使在没有参数的情况下,Lambda 表达式也可以有复杂的修饰符和属性,这些修饰符和属性的存在并不影响 Lambda 表达式的语义。因此,没有必要强制要求显式写出空的 ()

此外,提案还考虑了与 C++20 的兼容性。虽然 C++23 允许省略空的 (),但 C++20 中的代码仍然可以在 C++23 中正常工作,因为 C++23 的语法设计向后兼容。

三、编译器支持:主流编译器的跟进

随着 C++23 标准的推进,主流编译器已经对这一特性提供了支持。以下是各编译器的支持情况:

  • GCC:从 11 版本开始支持。
  • Clang:从 13 版本开始支持。
  • MSVC:从 19.44 版本开始支持。

这意味着开发者可以在这些编译器的 C++23 模式下开始使用这一新特性,享受更简洁的 Lambda 表达式语法。

四、对 C++ 编程的影响:简化语法与提升一致性

这一变化对 C++ 编程产生了多方面的积极影响,主要体现在以下几个方面:

4.1 简化语法

最直接的影响是简化了 Lambda 表达式的语法。开发者不再需要记住哪些情况下必须写空的 (),从而减少了语法错误。例如,以下代码在 C++23 中可以正常工作,而在 C++20 中则会报错:

auto l = [] mutable {};

4.2 提升语言一致性

这一改进还提高了语言的一致性。在 C++20 中,Lambda 表达式的语法规则较为复杂,不同情况下对 () 的要求不同。而在 C++23 中,无论 Lambda 表达式是否包含修饰符、属性或其他特性,都可以选择不写空的 (),这使得语法更加统一。

4.3 与其他新特性的结合

这一改进也为 Lambda 表达式与其他新特性的结合使用提供了便利。例如,与 [[nodiscard]] 属性结合,可以更灵活地表达函数调用的语义。

4.4 示例代码

以下是一些在 C++23 中使用省略 () 的 Lambda 表达式的示例代码:

#include <iostream>

int main() {
    // 简单的可变 Lambda
    auto fn = [x = 0] mutable {
        return x++;
    };
    std::cout << fn() << fn();

    // 带属性的 Lambda
    auto lm = [][[nodiscard]]()->int { return 42; };
    lm(); // 如果忽略返回值,可能会产生警告

    // 带模板参数的 Lambda
    auto templatedLambda = []<typename T>(T t) { return t; };
    std::cout << templatedLambda(42) << templatedLambda("Hello");

    return 0;
}

五、总结:更简洁、更一致的 Lambda 表达式

C++23 通过 P1102R2 提案,让 Lambda 表达式中的 () 在更多情况下可选,这一改进不仅简化了语法,还提高了语言的一致性和易用性。随着主流编译器对这一特性的支持,开发者可以在 C++23 项目中开始使用这一新特性,享受更简洁、更灵活的 Lambda 表达式。

这一变化是 C++ 语言持续演进的一个缩影,它展示了 C++ 社区在不断努力改进语言,使其更加现代化、简洁和易用。对于 C++ 开发者来说,这是一个值得期待和学习的新特性,它将为日常编程带来更多的便利和灵活性。

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

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

相关文章

在Qt中验证LDAP账户(Windows平台)

一、前言 原本以为在Qt&#xff08;Windows平台&#xff09;中验证 LDAP 账户很简单&#xff1a;集成Open LDAP的开发库即可。结果临了才发现&#xff0c;Open LDAP压根儿不支持Windows平台。沿着重用的原则&#xff0c;考虑迁移Open LDAP的源代码&#xff0c;却发现工作量不小…

【sylar-webserver】重构日志系统

文章目录 主要工作流程图FiberConditionBufferBufferManagerLogEvent 序列化 & 反序列化LoggerRotatingFileLogAppender 主要工作 实现&#xff0c; LogEvent 序列化和反序列化 &#xff08;使用序列化是为了更标准&#xff0c;如果转成最终的日志格式再存储&#xff08;确…

树莓派超全系列教程文档--(38)config.txt视频配置

config.txt视频配置 视频选项HDMI模式树莓派4-系列的HDMI树莓派5-系列的HDMI 复合视频模式enable_tvout LCD显示器和触摸屏ignore_lcddisable_touchscreen 通用显示选项disable_fw_kms_setup 文章来源&#xff1a; http://raspberry.dns8844.cn/documentation 原文网址 视频选…

线性DP:最短编辑距离

Dp 状态表示 f&#xff08;i&#xff0c;j&#xff09; 集合所有将A[1~i]变成B[1~j]的操作方式属性min 状态计算 &#xff08;划分&#xff09; 增f(i,j)f(i,j-1)1//A[i]元素要增加&#xff0c;说明A前i位置与B前j-1相同删f(i,j)f(i-1,j)1//A[i]元素要删除&#xff0c;说明A前i…

STM32——新建工程并使用寄存器以及库函数进行点灯

本文是根据江协科技提供的教学视频所写&#xff0c;旨在便于日后复习&#xff0c;同时供学习嵌入式的朋友们参考&#xff0c;文中涉及到的所有资料也均来源于江协科技&#xff08;资料下载&#xff09;。 新建工程并使用寄存器以及库函数进行点灯操作 新建工程步骤1.建立工程2.…

java集合框架day1————集合体系介绍

在进入正文之前&#xff0c;我们先来思考一下之前学过的数组有什么缺点&#xff1f; <1>长度开始时必须指定&#xff0c;而且一旦指定&#xff0c;不能更改 <2>保存的必须为同一类型的元素 <3>使用数组进行增加/删除元素的代码比较麻烦 为了方便读者理解&…

百度热力图数据获取,原理,处理及论文应用18

目录 0、数据简介0、示例数据1、百度热力图数据日期如何选择1.1、其他实验数据的时间1.2、看日历天气 2、百度热力图几天够研究&#xff1f;部分文章统计3、数据原理3.1 Bd09mc即百度墨卡托投影坐标系200单位的距离是可以自己设置的吗&#xff1f;3.2 csv文件字段说明3.3 ** 这…

【身份证扫描件识别表格】如何识别大量身份证扫描件将内容导出保存到Excel表格,一次性处理多张身份证图片导出Excel表格,基于WPF和腾讯云的实现方案

基于WPF和腾讯云的身份证扫描件批量处理方案 适用场景 本方案适用于需要批量处理大量身份证扫描件的场景,例如: 企业人事部门批量录入新员工身份信息银行或金融机构办理批量开户业务教育机构收集学生身份信息政府部门进行人口信息统计酒店、医院等需要实名登记的场所这些场景…

基于语义网络表示的不确定性推理

前文我们已经了解了: 1.不确定与非单调推理的基本概念:不确定与非单调推理的基本概念-CSDN博客 2.不确定与非单调推理的概率方法:不确定与非单调推理的概率方法-CSDN博客 3.不确定与非单调推理的可信度方法:不确定与非单调推理的可信度方法-CSDN博客 4.不确定与非单调推…

ICMAN防水触摸芯片 - 复杂环境下精准交互,提升触控体验

▍核心优势 ◆ 超强抗干扰能力 ◆ 工业级设计&#xff0c;一致性和稳定性好 ▍提供场景化解决方案 【智能厨电矩阵】抽油烟机档位调节 | 电磁炉火力触控 | 洗碗机模式切换 【卫浴设备方案】淋浴房雾化玻璃控制 | 智能马桶触控面板 | 浴缸水位感应 【工业控制应用】仪器仪…

【java实现+4种变体完整例子】排序算法中【希尔排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格

以下是希尔排序的详细解析&#xff0c;包含基础实现、常见变体的完整代码示例&#xff0c;以及各变体的对比表格&#xff1a; 一、希尔排序基础实现 原理 希尔排序是插入排序的改进版本&#xff0c;通过分步缩小增量间隔&#xff0c;将数组分成多个子序列进行插入排序&#…

matlab 处理海洋数据并画图的工具包--ocean_data_tools

matlab 处理海洋数据并画图的工具包–ocean_data_tools matlab 处理海洋数据并画图的工具包–ocean_data_tools ocean_data_tools 简化了提取、格式化和可视化免费可用的海洋学数据的过程。虽然可以在线访问大量海洋学数据&#xff0c;但由于获取这些数据并将其格式化为可用数据…

MCP:AI时代的“万能插座”,开启大模型无限可能

摘要&#xff1a;Model Context Protocol&#xff08;MCP&#xff09;由Anthropic在2024年底开源&#xff0c;旨在统一大模型与外部工具、数据源的通信标准。采用客户端-服务器架构&#xff0c;基于JSON-RPC 2.0协议&#xff0c;支持stdio、SSE、Streamable HTTP等多种通信方式…

静态网页的开发

文章目录 基于 idea 开发静态网页添加web框架前端配置服务器并启动服务资源名字不是 index 静态网页 流转 基于 idea 开发静态网页 添加web框架 方法1 方法2 前端 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8&quo…

【CPU】结合RISC-V CPU架构回答中断系统的7个问题(个人草稿)

结合RISC-V CPU架构对中断系统七个关键问题的详细解析&#xff0c;按照由浅入深的结构进行说明&#xff1a; 一、中断请求机制&#xff08;问题①&#xff09; 硬件基础&#xff1a; RISC-V通过CLINT&#xff08;Core Local Interrupter&#xff09;和PLIC&#xff08;Platfor…

uCOS3实时操作系统(任务切换和任务API函数)

文章目录 任务切换任务API函数 任务切换 C/OS-III 将 PendSV 的中断优先级配置为最低的中断优先级&#xff0c;这么一来&#xff0c; PendSV 异常的中断服务函数就会在其他所有中断处理完成后才被执行。C/OS-III 就是将任务切换的过程放到 PendSV 异常的中断服务函数中处理的。…

科学养生指南:解锁健康生活新方式

在快节奏的现代生活中&#xff0c;健康养生已成为人们关注的焦点。科学合理的养生方式&#xff0c;能帮助我们增强体质、预防疾病&#xff0c;享受更优质的生活。​ 饮食是健康养生的基石。遵循 “均衡饮食” 原则&#xff0c;每日饮食需包含谷类、蔬菜水果、优质蛋白质和健康…

第十四届蓝桥杯 2023 C/C++组 有奖问答

目录 题目&#xff1a; 题目描述&#xff1a; 题目链接&#xff1a; 思路&#xff1a; 核心思路&#xff1a; 思路详解&#xff1a; 代码&#xff1a; 代码详解&#xff1a; 题目&#xff1a; 题目描述&#xff1a; 题目链接&#xff1a; 蓝桥云课 有奖问答 思路&…

transformer注意力机制

单头注意力机制 import torch import torch.nn.functional as Fdef scaled_dot_product_attention(Q, K, V):# Q: (batch_size, seq_len, d_k)# K: (batch_size, seq_len, d_k)# V: (batch_size, seq_len, d_v)batch_size: 一次输入的句子数。 seq_len: 每个句子的词数。 d_mo…

QT 5.15 程序打包

说明&#xff1a; windeployqt 是 Qt 提供的一个工具&#xff0c;用于自动收集并复制运行 Qt 应用程序所需的动态链接库&#xff08;.dll 文件&#xff09;及其他资源&#xff08;如插件、QML 模块等&#xff09;到可执行文件所在的目录。这样你就可以将应用程序和这些依赖项一…