std::is_constant_evaluated

来自cppreference.com
< cpp‎ | types
 
 
 
在标头 <type_traits> 定义
constexprbool is_constant_evaluated()noexcept;
(C++20 起)

检查函数调用是否出现在常量求值的场合。若对调用的求值出现在明显常量求值的表达式或类型转换的求值中,则返回 true,否则返回 false

为确定下列变量的初始化式是否是明显常量求值,编译期可能首先试探性常量求值:

  • 引用类型或 const 限定的整数或枚举类型的变量
  • 静态及线程局域变量

不建议依赖于这种情况的结果。

int y =0;constint a = std::is_constant_evaluated()? y :1;// 试探性常量求值失败,常量求值被舍弃。// 变量 a 动态初始化为 1   constint b = std::is_constant_evaluated()?2: y;// 常量求值(std::is_constant_evaluation() == true)成功。// 变量 b 静态初始化为 2

目录

[编辑]参数

(无)

[编辑]返回值

若调用的求值出现在明显常量求值的表达式或类型转换的求值中,则返回 true,否则返回 false

[编辑]可能的实现

// 此实现要求 C++23 if consteval。constexprbool is_constant_evaluated()noexcept{if consteval {returntrue;}else{returnfalse;}}

[编辑]注解

std::is_constant_evaluated() 直接用作 static_assert 声明和 constexpr if 语句的条件时,返回值总是 true

由于 if consteval 不在 C++20 中,is_constant_evaluated 常由编译器扩展实现。

功能特性测试标准功能特性
__cpp_lib_is_constant_evaluated201811L(C++20)std::is_constant_evaluated

[编辑]示例

#include <cmath>#include <iostream>#include <type_traits>   constexprdouble power(double b, int x){if(std::is_constant_evaluated()&&!(b ==0.0&& x <0)){// 常量求值语境:使用 constexpr 友好的算法。if(x ==0)return1.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{// 令代码生成器生成。returnstd::pow(b, double(x));}}   int main(){// 常量表达式语境constexprdouble 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) 断言变量拥有静态初始化,即零初始化常量初始化[编辑]
close