C++23 中 constexpr 的重要改动

news2025/7/31 19:14:06

文章目录

    • 1. constexpr 函数中使用非字面量变量、标号和 goto (P2242R3)
      • 示例代码
    • 2. 允许 constexpr 函数中的常量表达式中使用 static 和 thread_local 变量 (P2647R1)
      • 示例代码
    • 3. constexpr 函数的返回类型和形参类型不必为字面类型 (P2448R2)
      • 示例代码
    • 4. 不存在满足核心常量表达式要求的调用的 constexpr 函数 (P2448R2)
      • 示例代码
    • 总结表格

在 C++23 标准中, constexpr 特性迎来了一系列令人瞩目的改动,这些改动进一步提升了 C++ 的编译时计算能力和代码的灵活性。下面我们将详细介绍这些改动,并通过表格的形式进行总结。

1. constexpr 函数中使用非字面量变量、标号和 goto (P2242R3)

在 C++23 之前,constexpr 函数的使用受到较多限制,不能在其中使用非字面量变量、标号和 goto 语句。但在 C++23 中,这些限制被放宽了。这意味着在 constexpr 函数里,我们可以更自由地编写代码,利用非字面量变量进行计算,使用标号和 goto 语句实现复杂的控制流。

在过去,由于这些限制,一些看似合理的代码可能会被编译器拒绝。例如下面的代码:

template<typename T> constexpr bool f() {
  if (std::is_constant_evaluated()) {
    // ...
    return true;
  } else {
    T t;
    // ...
    return true;
  }
}
struct nonliteral { nonliteral(); };
static_assert(f<nonliteral>());

在之前的标准中,这段代码可能会因为 nonliteral 是一个非字面类型而导致编译失败,尽管导致失败的那一行代码并不在常量求值的上下文中。而在 C++23 中,这样的代码是有效的。

从编译器支持情况来看,GCC 12 和 Clang 15 开始支持这一改动。这一改动的原理是,只要这些非字面量变量、标号和 goto 语句在编译时不被求值,它们在函数中的存在就不会有问题。因为 constexpr 函数可能在编译时求值,也可能在运行时求值。如果我们想在 constexpr 函数中调用一段保证在编译时求值的代码,需要将这段代码放在 if constevalif (std::is_constant_evaluated()) 条件下的代码块中。

示例代码

#include <iostream>

constexpr int func(int x) {
    int result = 0;
    if (x > 0) {
        result = x * 2;
    } else {
        // 使用标号和 goto
        label:
        result = -x;
    }
    return result;
}

int main() {
    constexpr int value = func(5);
    std::cout << "Result: " << value << std::endl;
    return 0;
}

2. 允许 constexpr 函数中的常量表达式中使用 static 和 thread_local 变量 (P2647R1)

在 C++23 之前,constexpr 函数的常量表达式中不允许使用 staticthread_local 变量。C++23 打破了这个限制,允许在 constexpr 函数的常量表达式中使用这两种变量。这为编译时计算提供了更多的可能性,例如可以在编译时初始化一些静态变量或线程局部变量。

最初,constexpr 函数中根本不允许声明任何 static 变量。后来在 [P2242R3] 中有所放宽,规则改为控制流不能经过 static 变量的初始化。对于 static(或者更糟糕的 thread_local)变量,其初始化器可能会运行任意代码,所以之前有这样的限制是合理的。但对于 static constexpr 变量,根据定义,它必须是常量初始化的,不存在何时运行初始化的问题,它就是一个常量。

下面我们来看一个例子:

char xdigit(int n) {
    static constexpr char digits[] = "0123456789abcdef";
    return digits[n];
}

这个函数原本是完全没问题的,但当我们尝试将其扩展为在编译时也能工作时,就会遇到问题:

constexpr char xdigit(int n) {
    static constexpr char digits[] = "0123456789abcdef";
    return digits[n];
}

在之前的标准中,这段代码是格式错误的,但在 C++23 中,它是有效的。

在之前为了实现类似的功能,有几种变通方法,但都有各自的缺点。比如可以完全避开 static 变量,直接索引字面量,但这只在我们只需要使用一次时才有效:

constexpr char xdigit(int n) {
    return "0123456789abcdef"[n];
}

也可以将 static 变量移到非局部作用域,但我们希望将其设为局部变量是有原因的,它只与这个特定的函数相关:

static constexpr char digits[] = "0123456789abcdef";
constexpr char xdigit(int n) {
    return digits[n];
}

还可以将变量设为非 static,但编译器很难对其进行优化,会导致代码生成效果变差:

constexpr char xdigit(int n) {
    constexpr char digits[] = "0123456789abcdef";
    return digits[n];
}

示例代码

#include <iostream>

constexpr int func() {
    static int counter = 0;
    counter++;
    return counter;
}

int main() {
    constexpr int value = func();
    std::cout << "Counter value: " << value << std::endl;
    return 0;
}

3. constexpr 函数的返回类型和形参类型不必为字面类型 (P2448R2)

在 C++23 之前,constexpr 函数的返回类型和形参类型必须是字面类型。C++23 放宽了这一要求,允许 constexpr 函数的返回类型和形参类型不必为字面类型。这使得 constexpr 函数的使用更加灵活,可以处理更多类型的数据。

示例代码

#include <iostream>
#include <string>

constexpr std::string func(const std::string& str) {
    return str + " appended";
}

int main() {
    constexpr std::string result = func("Hello");
    std::cout << "Result: " << result << std::endl;
    return 0;
}

4. 不存在满足核心常量表达式要求的调用的 constexpr 函数 (P2448R2)

在 C++23 中,对于那些不存在满足核心常量表达式要求的调用的 constexpr 函数,也有了新的处理方式。这使得在某些情况下,即使函数的调用不满足核心常量表达式的要求,函数仍然可以作为 constexpr 函数存在。

示例代码

#include <iostream>

constexpr int func(int x) {
    if (x > 0) {
        return x * 2;
    } else {
        // 这里的调用可能不满足核心常量表达式要求
        return -x;
    }
}

int main() {
    int value = 5;
    // 这里的调用可能不是常量表达式
    int result = func(value);
    std::cout << "Result: " << result << std::endl;
    return 0;
}

总结表格

改动内容提案编号说明
constexpr 函数中使用非字面量变量、标号和 gotoP2242R3放宽了 constexpr 函数的使用限制,允许使用非字面量变量、标号和 goto 语句,只要在编译时不被求值即可,GCC 12 和 Clang 15 开始支持
允许 constexpr 函数中的常量表达式中使用 static 和 thread_local 变量P2647R1打破了 constexpr 函数常量表达式中对 staticthread_local 变量的限制,之前 static 变量相关规则在 [P2242R3] 中有所调整,现在 static constexpr 变量在 constexpr 函数中使用更合理
constexpr 函数的返回类型和形参类型不必为字面类型P2448R2使 constexpr 函数的使用更加灵活,可处理更多类型的数据
不存在满足核心常量表达式要求的调用的 constexpr 函数P2448R2对于不满足核心常量表达式要求的调用的 constexpr 函数有了新的处理方式

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

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

相关文章

全面解析React内存泄漏:原因、解决方案与最佳实践

在开发React应用时&#xff0c;内存泄漏是一个常见但容易被忽视的问题。如果处理不当&#xff0c;它会导致应用性能下降、卡顿甚至崩溃。由于React的组件化特性&#xff0c;许多开发者可能没有意识到某些操作&#xff08;如事件监听、异步请求、定时器等&#xff09;在组件卸载…

【FreeRTOS】事件标志组

文章目录 1 简介1.1事件标志1.2事件组 2事件标志组API2.1创建动态创建静态创建 2.2 删除事件标志组2.3 等待事件标志位2.4 设置事件标志位在任务中在中断中 2.5 清除事件标志位在任务中在中断中 2.6 获取事件组中的事件标志位在任务中在中断中 2.7 函数xEventGroupSync 3 事件标…

超级扩音器手机版:随时随地,大声说话

在日常生活中&#xff0c;我们常常会遇到手机音量太小的问题&#xff0c;尤其是在嘈杂的环境中&#xff0c;如KTV、派对或户外活动时&#xff0c;手机自带的音量往往难以满足需求。今天&#xff0c;我们要介绍的 超级扩音器手机版&#xff0c;就是这样一款由上海聚告德业文化发…

【数据可视化-27】全球网络安全威胁数据可视化分析(2015-2024)

&#x1f9d1; 博主简介&#xff1a;曾任某智慧城市类企业算法总监&#xff0c;目前在美国市场的物流公司从事高级算法工程师一职&#xff0c;深耕人工智能领域&#xff0c;精通python数据挖掘、可视化、机器学习等&#xff0c;发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…

【6G 开发】NV NGC

配置 生成密钥 API Keys 生成您自己的 API 密钥&#xff0c;以便通过 Docker 客户端或通过 NGC CLI 使用 Secrets Manager、NGC Catalog 和 Private Registry 的 NGC 服务 以下个人 API 密钥已成功生成&#xff0c;可供此组织使用。这是唯一一次显示您的密钥。 请妥善保管您的…

SIEMENS PLC程序解读 -Serialize(序列化)SCATTER_BLK(数据分散)

1、程序数据 第12个字节 PI 2、程序数据 第16个字节 PI 3、程序数据 第76个字节 PO 4、程序代码 2、程序解读 图中代码为 PLC 梯形图&#xff0c;主要包含以下指令及功能&#xff1a; Serialize&#xff08;序列化&#xff09;&#xff1a; 将 SRC_VARIABLE&#xff…

宁德时代25年时代长安动力电池社招入职测评SHL题库Verify测评语言理解数字推理真题

测试分为语言和数字两部分&#xff0c;测试时间各为17分钟&#xff0c;测试正式开始后不能中断或暂停

【硬核解析:基于Python与SAE J1939-71协议的重型汽车CAN报文解析工具开发实战】

引言&#xff1a;重型汽车CAN总线的数据价值与挑战 随着汽车电子化程度的提升&#xff0c;控制器局域网&#xff08;CAN总线&#xff09;已成为重型汽车的核心通信网络。不同控制单元&#xff08;ECU&#xff09;通过CAN总线实时交互海量报文数据&#xff0c;这些数据隐藏着车…

Uniapp 自定义 Tabbar 实现教程

Uniapp 自定义 Tabbar 实现教程 1. 简介2. 实现步骤2.1 创建自定义 Tabbar 组件2.2 配置 pages.json2.3 在 App.vue 中引入组件 3. 实现过程中的关键点3.1 路由映射3.2 样式设计3.3 图标处理 4. 常见问题及解决方案4.1 页面跳转问题4.2 样式适配问题4.3 性能优化 5. 扩展功能5.…

记录一次使用面向对象的C语言封装步进电机驱动

简介 (2025/4/21) 本库对目前仅针对TB6600驱动下的42步进电机的基础功能进行了一定的封装, 也是我初次尝试以面向对象的思想去编写嵌入式代码, 和直流电机的驱动步骤相似在调用stepmotor_attach()函数和stepmotor_init()函数之后仅通过结构体数组stepm然后指定枚举变量中的id即…

Spark-streaming核心编程

1.导入依赖‌&#xff1a; <dependency> <groupId>org.apache.spark</groupId> <artifactId>spark-streaming-kafka-0-10_2.12</artifactId> <version>3.0.0</version> </dependency> 2.编写代码‌&#xff1a; 创建Sp…

vue3+TS+echarts 折线图

需要实现的效果如下 <script setup lang"ts" name"RepsSingleLineChart">import * as echarts from echartsimport { getInitecharts } from /utils/echartimport type { EChartsOption } from echarts// 定义 props 类型interface Props {id: strin…

小火电视桌面TV版下载-小火桌面纯净版下载-官方历史版本安装包

别再费心地寻找小火桌面的官方历史版本安装包啦&#xff0c;试试乐看家桌面吧&#xff0c;它作为纯净版本的第三方桌面&#xff0c;具有诸多优点。 界面简洁纯净&#xff1a;乐看家桌面设计简洁流畅&#xff0c;页面简洁、纯净无广告&#xff0c;为用户打造了一个干净的电视操…

androidstudio安装配置

B站配置视频AndroidStudio安装配置教程&#xff08;最新版本教程&#xff09;3分钟搞定 快速安装使用_哔哩哔哩_bilibili 1、环境变量 D:\AndroidSdk ANDROID_HOME ANDROID_SDK_HOME 2、新建 3、配置 distributionUrlhttps://mirrors.cloud.tencent.com/gradle/gradle-8.11.1-…

《AI大模型趣味实战》基于RAG向量数据库的知识库AI问答助手设计与实现

基于RAG向量数据库的知识库AI问答助手设计与实现 引言 随着大语言模型&#xff08;LLM&#xff09;技术的快速发展&#xff0c;构建本地知识库AI问答助手已成为许多企业级应用的需求。本研究报告将详细介绍如何基于FLASK开发一个使用本地OLLAMA大模型底座的知识库AI问答助手&…

BeeWorks Meet:私有化部署视频会议的高效选择

在数字化时代&#xff0c;视频会议已成为企业沟通协作的重要工具。然而&#xff0c;对于金融、政务、医疗等对数据安全和隐私保护要求极高的行业来说&#xff0c;传统的公有云视频会议解决方案往往难以满足其严格的安全标准。此时&#xff0c;BeeWorks Meet 私有化部署视频会议…

IPv6 技术细节 | 源 IP 地址选择 / Anycast / 地址自动配置 / 地址聚类分配

注&#xff1a;本文为 “IPv6 技术细节” 相关文章合集。 部分文章中提到的其他文章&#xff0c;一并引入。 略作重排&#xff0c;未整理去重。 如有内容异常&#xff0c;请看原文。 闲谈 IPv6 - 典型特征的一些技术细节 iteye_21199 于 2012-11-10 20:54:00 发布 0. 巨大的…

【工具】使用 MCP Inspector 调试服务的完全指南

Model Context Protocol (MCP) Inspector 是一个交互式开发工具&#xff0c;专为测试和调试 MCP 服务器而设计。本文将详细介绍如何使用 Inspector 工具有效地调试和测试 MCP 服务。 1. MCP Inspector 简介 MCP Inspector 提供了直观的界面&#xff0c;让开发者能够&#xff…

【音视频】AVIO输入模式

内存IO模式 AVIOContext *avio_alloc_context( unsigned char *buffer, int buffer_size, int write_flag, void *opaque, int (*read_packet)(void *opaque, uint8_t *buf, int buf_size), int (*write_packet)(void *opaque, uint8_t *buf, int buf_size), int64_t (*seek)(…

Uniapp:scroll-view(区域滑动视图)

目录 一、基本概述二、属性说明三、基本使用3.1 纵向滚动3.2 横向滚动一、基本概述 scroll-view,可滚动视图区域。用于区域滚动。 二、属性说明 属性名类型默认值说明平台差异说明scroll-xBooleanfalse允许横向滚动scroll-yBooleanfalse允许纵向滚动三、基本使用 3.1 纵向滚…