Namespaces
Variants

std:: is_constant_evaluated

From cppreference.net
Utilities library
定义于头文件 <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_evaluated() == true 时常量求值成功
// 变量 b 通过静态初始化赋值为 2

目录

参数

(无)

返回值

若调用的求值发生在明显常量求值的表达式或转换的求值过程中,则为 true ;否则为 false

可能的实现

// 此实现需要 C++23 的 if consteval 特性
constexpr bool is_constant_evaluated() noexcept
{
    if consteval
    {
        return true;
    }
    else 
    {
        return false;
    }
}

注释

当直接用作 static_assert 声明或 constexpr if语句 的条件时, std :: is_constant_evaluated ( ) 始终返回 true

由于 C++20 中缺少 if consteval std::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 specifier (C++11) 指定变量或函数的值可在编译时计算
consteval specifier (C++20) 指定函数为 立即函数 ,即对该函数的每次调用都必须在常量求值中进行
constinit specifier (C++20) 断言变量具有静态初始化,即 零初始化 常量初始化