Namespaces
Variants

std:: is_constant_evaluated

From cppreference.net
Utilities library
Определено в заголовочном файле <type_traits>
constexpr bool is_constant_evaluated ( ) noexcept ;
(начиная с C++20)

Определяет, происходит ли вызов функции в контексте константного вычисления. Возвращает true если вычисление вызова происходит в процессе вычисления выражения или преобразования, которое является явно константно вычисляемым ; в противном случае возвращает false .

Чтобы определить, являются ли инициализаторы следующих переменных явно константно-вычисляемыми, компиляторы могут сначала выполнить пробное константное вычисление:

  • переменные с ссылочным типом или const-квалифицированным целочисленным или перечислимым типом;
  • статические и thread_local переменные.

В данном случае не рекомендуется полагаться на результат.

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 statement , std :: is_constant_evaluated ( ) всегда возвращает true .

Поскольку if consteval отсутствует в C++20, 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 не может быть преобразовано в rvalue
    // в контексте константного выражения
    // Эквивалентно 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) утверждает, что переменная имеет статическую инициализацию, т.е. нулевую инициализацию и константную инициализацию