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

原著在线阅读地址

相关推荐
2301_821369612 小时前
嵌入式实时C++编程
开发语言·c++·算法
sjjhd6522 小时前
多核并行计算优化
开发语言·c++·算法
田野追逐星光3 小时前
STL的容器vector的模拟实现
开发语言·c++
Tansmjs3 小时前
实时数据可视化库
开发语言·c++·算法
WBluuue3 小时前
Codeforces 1075 Div2(ABC1C2D1D2)
c++·算法
添砖java‘’3 小时前
线程的互斥与同步
linux·c++·操作系统·线程·信息与通信
2401_838472513 小时前
C++模拟器开发实践
开发语言·c++·算法
3108748764 小时前
0005.C/C++学习笔记5
c语言·c++·学习
s1hiyu4 小时前
实时控制系统验证
开发语言·c++·算法