【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函数可以用在一个作用域更广的语境中。

原著在线阅读地址

相关推荐
总斯霖22 分钟前
P15445永远在一起!题解(月赛T2)
数据结构·c++·算法·深度优先
Yupureki1 小时前
《C++实战项目-高并发内存池》4.CentralCache构造
c语言·开发语言·c++·单例模式·github
样例过了就是过了1 小时前
LeetCode热题100 全排列
数据结构·c++·算法·leetcode·dfs
2401_898075121 小时前
分布式系统监控工具
开发语言·c++·算法
OKkankan1 小时前
撕 STL 系列:封装红黑树实现 mymap 和 myset
java·c++·算法
ab1515171 小时前
3.15二刷基础90、105、106、110
数据结构·c++·算法
寻寻觅觅☆1 小时前
东华OJ-进阶题-10-分解质因数(C++)
数据结构·c++·算法
Darkwanderor2 小时前
数据结构——ST表和RMQ问题
数据结构·c++·动态规划·st表·rmq问题
是梦终空1162 小时前
模板编译期机器学习
开发语言·c++·算法
艾莉丝努力练剑2 小时前
文件描述符fd:跨进程共享机制
java·linux·运维·服务器·开发语言·c++