std::is_constant_evaluated

来自cppreference.com
< cpp‎ | types
 
 
工具库
语言支持
类型支持(基本类型、RTTI)
库功能特性测试宏 (C++20)
动态内存管理
程序工具
协程支持 (C++20)
变参数函数
is_constant_evaluated
(C++20)
调试支持
(C++26)
三路比较
(C++20)
(C++20)(C++20)(C++20)
(C++20)(C++20)(C++20)
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)   
(C++20)
交换类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)

 
在标头 <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) 断言变量拥有静态初始化,即零初始化常量初始化