std::is_constant_evaluated
来自cppreference.com
在标头 <type_traits> 定义
|
||
constexpr bool is_constant_evaluated() noexcept; |
(C++20 起) | |
检查函数调用是否出现在常量求值的场合。若对调用的求值出现在明显常量求值的表达式或类型转换的求值中,则返回 true,否则返回 false。
为确定下列变量的初始化式是否是明显常量求值,编译期可能首先试探性常量求值:
- 引用类型或 const 限定的整数或枚举类型的变量
- 静态及线程局域变量
不建议依赖于这种情况的结果。
int y = 0; const int a = std::is_constant_evaluated() ? y : 1; // 试探性常量求值失败,常量求值被舍弃。 // 变量 a 动态初始化为 1 const int b = std::is_constant_evaluated() ? 2 : y; // 常量求值(std::is_constant_evaluation() == true)成功。 // 变量 b 静态初始化为 2
参数
(无)
返回值
若调用的求值出现在明显常量求值的表达式或类型转换的求值中,则返回 true,否则返回 false。
可能的实现
// 此实现要求 C++23 if consteval。 constexpr bool is_constant_evaluated() noexcept { if consteval { return true; } else { return false; } } |
注解
std::is_constant_evaluated() 直接用作 static_assert 声明和 constexpr if 语句的条件时,返回值总是 true。
由于 if consteval 不在 C++20 中,is_constant_evaluated
常由编译器扩展实现。
功能特性测试宏 | 值 | 标准 | 功能特性 |
---|---|---|---|
__cpp_lib_is_constant_evaluated |
201811L | (C++20) | std::is_constant_evaluated
|
示例
运行此代码
#include <cmath> #include <iostream> #include <type_traits> constexpr double power(double b, int x) { if (std::is_constant_evaluated() && !(b == 0.0 && x < 0)) { // 常量求值语境:使用 constexpr 友好的算法。 if (x == 0) return 1.0; double r {1.0}; double p {x > 0 ? b : 1.0 / b}; for (auto u = unsigned(x > 0 ? x : -x); u != 0; u /= 2) { if (u & 1) r *= p; p *= p; } return r; } else { // 令代码生成器生成。 return std::pow(b, double(x)); } } int main() { // 常量表达式语境 constexpr double kilo = power(10.0, 3); int n = 3; // 非常量表达式,因为 n 不能在常量表达式语境中转换成右值 // 等价于 std::pow(10.0, double(n)) double mucho = power(10.0, n); std::cout << kilo << " " << mucho << "\n"; // (3) }
输出:
1000 1000
参阅
constexpr 说明符(C++11)
|
指定变量或函数的值能在编译时计算 |
consteval 说明符(C++20)
|
指定函数为立即函数,即对该函数的每次调用必须在常量求值中进行 |
constinit 说明符(C++20)
|
断言变量拥有静态初始化,即零初始化与常量初始化 |