0

0

c++ constexpr关键字_c++编译期常量优化

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-12 08:44:02

|

623人浏览过

|

来源于php中文网

原创

constexpr函数仅在需常量表达式时编译期求值,否则退化为运行时调用;其体内容纳操作受限,变量初始化须为常量表达式且类型为字面量类型,不可与const混用。

c++ constexpr关键字_c++编译期常量优化

constexpr 函数必须满足“编译期可求值”约束

不是所有带 constexpr 的函数都能在编译期运行。编译器只在上下文明确要求常量表达式(比如数组长度、模板非类型参数、static_assert 条件)时,才尝试对 constexpr 函数做编译期求值。否则它可能退化为普通函数,在运行时调用。

常见错误是误以为加了 constexpr 就自动“优化掉了”,结果发现生成的汇编里仍有函数调用指令。根本原因是函数体用了不被允许的操作:比如堆分配、虚函数调用、std::vectordynamic_cast,甚至未初始化的局部变量读取——这些都会让函数失去 constexpr 资格(C++14 起放宽了部分限制,但仍禁止 I/O 和全局状态修改)。

  • 检查是否触发了 constexpr 求值:用 static_assert(func(5) == 25, "");若报错,说明该调用无法在编译期完成
  • C++17 起支持 if constexpr,可在编译期分支中安全使用 constexpr 表达式,但分支条件本身必须是常量表达式
  • 避免在 constexpr 函数中调用非 constexpr 标准库函数(如 std::sqrt 在 C++20 前不是 constexpr

constexpr 变量必须用常量表达式初始化

constexpr 变量本质是“命名的编译期常量”,不是“带编译期检查的 const 变量”。它要求初始化器必须是常量表达式,且类型需支持字面量类型(literal type):有平凡析构、所有构造函数和赋值操作符都是 constexpr,且成员也满足同样要求。

典型陷阱是把 constexprconst 混用。例如:

const int x = 42;  
constexpr int y = x; // ❌ 编译失败:x 不是常量表达式(虽值不变,但未标记 constexpr)
即使 x 值固定,它仍不具备编译期可求值性。正确写法是:
constexpr int x = 42;  
constexpr int y = x; // ✅

  • 类类型要支持 constexpr 构造,必须显式声明构造函数为 constexpr,且函数体为空或仅含常量表达式计算
  • constexpr 数组大小必须是常量表达式,int arr[func()]func() 必须是 constexpr 且实际在编译期被求值
  • 注意链接性:constexpr 全局变量默认是 inline(C++17 起),无需额外加 inlineextern

constexpr 与模板元编程的协作边界

constexpr 并不能替代所有模板元编程(TMP)。它擅长数值计算、字符串字面量处理(C++20)、简单数据结构构建(如 std::array 初始化),但遇到类型推导、SFINAE、偏特化等需求时,仍需传统 TMP。

OmniAudio
OmniAudio

OmniAudio 是一款通过 AI 支持将网页、Word 文档、Gmail 内容、文本片段、视频音频文件都转换为音频播客,并生成可在常见 Podcast ap

下载

立即学习C++免费学习笔记(深入)”;

例如,想根据整型值生成不同类型的容器,仅靠 constexpr 无法做到:

template  
using container_t = std::conditional_t<(N > 10), std::vector, std::array>;
这里 N 是非类型模板参数,属于编译期信息,但它的用途是控制类型选择,不是计算值——这是模板机制的职责,constexpr 插不上手。

  • constexpr 函数可用于生成模板实参(如 std::array),前提是 compute_size() 真正在编译期求值
  • C++20 引入 consteval,强制函数只能在编译期调用,适合封装纯编译期逻辑,避免意外运行时降级
  • 过度依赖 constexpr 实现复杂逻辑(如编译期 JSON 解析)会导致编译时间飙升,需权衡可维护性与构建性能

编译期优化效果取决于上下文和标准版本

同一个 constexpr 函数,在不同场景下是否真正消除运行时开销,要看调用点是否构成常量表达式,以及编译器是否启用对应 C++ 标准支持。GCC/Clang 对 C++20 constexpr 字符串和容器的支持仍有限,MSVC 更早落地部分特性。

例如,C++20 允许 constexpr std::string_viewstd::array 成员函数,但若用 GCC 11 编译,sv.substr(0, 3) 在编译期可能仍不被接受,导致代码无法通过 —— 这不是写法错,而是工具链能力边界。

  • -std=c++20(而非 c++17)才能启用多数现代 constexpr 扩展
  • 开启 -O2 或更高优化等级有助于编译器识别并内联 constexpr 函数,但不会改变其是否能在编译期求值的本质
  • 调试时(-O0)某些 constexpr 调用可能被迫转为运行时执行,别把它当成“优化开关”,它是语义约束
编译期常量的核心不在“快”,而在“确定性”——它把一部分计算从运行时契约转移到编译时契约。一旦写错约束条件,轻则编译失败,重则静默退化为运行时行为,而后者最难排查。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

408

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

532

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1463

2023.10.24

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

731

2023.08.22

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

75

2025.09.18

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号