Пространства имён
Варианты
Действия

std::is_constant_evaluated

Материал из cppreference.com
< cpp‎ | types
 
 
Библиотека утилит
Языковая поддержка
Поддержка типов (базовые типы, RTTI)
Макросы тестирования функциональности библиотеки (C++20)    
Управление динамической памятью
Программные утилиты
Поддержка сопрограмм(C++20)
Вариативные функции
is_constant_evaluated
(C++20)
Трёхстороннее сравнение (C++20)
(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++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>
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_evaluated() == true выполняется успешно.// Переменная b статически инициализируется значением 2

Содержание

[править]Параметры

(нет)

[править]Возвращаемое значение

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

[править]Возможная реализация

// Эта реализация требует C++23, если consteval.constexprbool is_constant_evaluated()noexcept{if consteval {returntrue;}else{returnfalse;}}

[править]Примечание

При непосредственном использовании в качестве условия объявления static_assert или оператора constexpr if, std::is_constant_evaluated() всегда возвращает true.

Поскольку if consteval отсутствует в C++20, std::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