【Effective Modern C++】第三章 转向现代C++:15. 尽可能使用constexpr

constexpr作用于对象时,是加强版的const;但作用于函数时含义截然不同 ------ 它并非简单限定 "返回 const 值" 或 "结果编译期可知",而是根据调用场景动态适配。

  • constexpr对象:必然具备const只读属性,且必须由编译期可知的值完成初始化(技术上是翻译期,包含编译 + 链接,编程层面可视为编译期)。

  • constexpr函数:不保证返回值是 const,也不保证结果一定编译期可知------ 结果是否编译期确定,完全取决于传入的实参:

    • 若实参是编译期已知的常量 → 结果在编译期计算,可用于 "整型常量表达式" 语境;
    • 若实参是运行期才能确定的值 → 函数退化为普通函数,在运行期计算结果。

编译期可知的常量整型值,可用于 C++ 要求 "整型常量表达式" 的核心语境:数组尺寸、整型模板实参(如std::array的长度)、枚举量的值、对齐修饰符(alignas)等。普通const对象无法满足这些场景(仅保证只读,不保证编译期初始化),因此:

所有constexpr对象都是const对象,但并非所有const对象都是constexpr对象。

constexpr函数的特性与限制:

  1. 适用语境:可同时适配 "编译期常量计算" 和 "运行期普通计算",无需为两种场景编写不同函数;

  2. 类型限制:仅能接收和返回字面值类型(即编译期可确定值的类型,如内置类型、符合条件的自定义类型);

  3. 实现限制(C++ 版本差异):

    • C++11:函数体仅允许 1 行 return 语句,需用三元运算符替代 if-else、递归替代循环;
    • C++14:限制大幅放宽,支持循环、局部变量、void 返回值,甚至允许修改对象状态的成员函数(如自定义类型的 setter)。

自定义类型对constexpr的支持:

  • C++11:仅构造函数、返回字面值类型的 const getter 可声明为constexpr(成员函数隐式 const);
  • C++14:可将修改对象状态的 setter 声明为constexpr(void 返回值也允许),能基于编译期对象完成更复杂的编译期计算(如 Point 类的中点、镜像点计算)。

注意:constexpr是对象 / 函数接口的核心组成部分------ 声明后若随意移除(如函数新增 I/O 语句),依赖它的客户端代码(如用在模板参数、数组大小)会直接编译失败,需长期遵守。

总结

  • constexpr对象都具备const属性,并由编译期已知的值完成初始化。
  • constexpr函数在调用时若传入的实参值是编译器已知的,则会产出编译期结果。
  • 比起非constexpr对象或constexpr函数而言,constexpr对象或是constexpr函数可以用在一个作用域更广的语境中。

原著在线阅读地址

相关推荐
特种加菲猫14 分钟前
从零开始手撕AVL树:详解插入、平衡因子更新与四种旋转
开发语言·c++
萑澈15 分钟前
算法竞赛入门:C++ STL核心用法与时空复杂度速查手册
数据结构·c++·算法·stl
江屿风1 小时前
C++OJ题经验总结(竞赛)1
开发语言·c++·笔记·算法
运筹vivo@1 小时前
LeetCode 2405. 子字符串的最优划分
c++·算法·leetcode·职场和发展·哈希表
有点。2 小时前
C++(枚举法一练习题)
开发语言·c++·算法
basketball6162 小时前
C++ 单例模式完全指南:从饿汉式到现代 C++ 的最佳实践
java·c++·单例模式
玖釉-2 小时前
栈——栈的定义及基本操作
c++·windows·算法·图形渲染
不想写代码的星星2 小时前
C++ 内存序六件套:从完全同步到爱咋咋地
c++
haibindev4 小时前
别让AI再从零写一堆优美的屎山了
c++·ai编程·claude·流媒体·codex·代码复用