std::is_constant_evaluated
来自cppreference.com
定义于头文件 <type_traits>
|
||
constexpr bool is_constant_evaluated() noexcept; |
(C++20 起) | |
检查函数调用是否出现在常量求值的场合。若对调用的求值出现在明显常量求值的表达式或类型转换的求值中,返回 true
,否则返回 false
。
以下表达式(包括到目标类型的隐式类型转换)是明显常量求值的:
- 语法上要求常量表达式的场合,包括
- 数组长度
- new 表达式中除第一维以外的数组长度
- 位域长度
- 枚举项初始化器
- 对齐
- case 表达式
- 非类型模板实参
- noexcept 规定
-
static_assert
声明 - 条件性
explicit
指定符
- constexpr if 语句的条件
- 立即调用
- 概念定义、嵌套要求和 requires 子句中的表达式
- 可用于常量表达式的变量的初始化器,包括
- constexpr 变量的初始化器
- 引用和带 const 限定的整数或枚举类型的变量的初始化器,若该初始化器为常量表达式
- 静态及线程局域变量的初始化器,若该初始化器的所有子表达式(含构造函数调用和隐式转换)都是常量表达式(即该初始化器为常量初始化器)
测试最后两个条件可能涉及对初始化器的试探性常量求值。不建议依赖此时的结果。
int y; 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
。
注意
std::is_constant_evaluated() 直接作为 static_assert
声明和 constexpr if 语句的条件时,返回值总是 true 。
示例
运行此代码
#include <type_traits> #include <cmath> #include <iostream> 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, p = x > 0 ? b : 1.0 / b; auto u = unsigned(x > 0 ? x : -x); while (u != 0) { if (u & 1) r *= p; u /= 2; 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