std:: is_constant_evaluated
|
Определено в заголовочном файле
<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)
|
утверждает, что переменная имеет статическую инициализацию, т.е. нулевую инициализацию и константную инициализацию |