std::is_constant_evaluated
| Definiert in der Kopfdatei <type_traits> |
||
| constexpr bool is_constant_evaluated() noexcept; |
(seit C++20) | |
Ermittelt, ob der Funktionsaufruf innerhalb eines konstant ausgewerteten Kontexts erfolgt. Gibt true zurück, wenn die Auswertung des Aufrufs innerhalb der Auswertung eines Ausdrucks oder einer Konvertierung erfolgt, der explizit konstant ausgewertet wird; andernfalls wird false zurückgegeben.
Um zu bestimmen, ob Initialisierer von folgenden Variablen explizit konstant ausgewertet werden, können Compiler zunächst eine Probekonstantenauswertung durchführen.
- Variablen mit Referenztyp oder const-qualifiziertem ganzzahligem oder Aufzählungstyp;
- statische und Thread-lokale Variablen.
Es wird nicht empfohlen, sich in diesem Fall auf das Ergebnis zu verlassen.
int y = 0; const int a = std::is_constant_evaluated() ? y : 1; // Trial constant evaluation fails. The constant evaluation is discarded. // Variable a is dynamically initialized with 1 const int b = std::is_constant_evaluated() ? 2 : y; // Constant evaluation with std::is_constant_evaluated() == true succeeds. // Variable b is statically initialized with 2
Inhalt |
[bearbeiten] Parameter
(keine)
[bearbeiten] Rückgabewert
true, wenn die Auswertung des Aufrufs innerhalb der Auswertung eines Ausdrucks oder einer Konvertierung erfolgt, der explizit konstant ausgewertet wird; andernfalls false.
[bearbeiten] Mögliche Implementierung
// This implementation requires C++23 if consteval. constexpr bool is_constant_evaluated() noexcept { if consteval { return true; } else { return false; } } |
[bearbeiten] Hinweise
Wenn std::is_constant_evaluated() direkt als Bedingung einer static_assert-Deklaration oder einer constexpr if-Anweisung verwendet wird, gibt std::is_constant_evaluated() immer true zurück.
Da if consteval in C++20 fehlt, wird std::is_constant_evaluated typischerweise mithilfe einer Compilererweiterung implementiert.
| Feature-Test-Makro | Wert | Std | Feature |
|---|---|---|---|
__cpp_lib_is_constant_evaluated |
201811L |
(C++20) | std::is_constant_evaluated
|
[bearbeiten] Beispiel
#include <cmath> #include <iostream> #include <type_traits> constexpr double power(double b, int x) { if (std::is_constant_evaluated() && !(b == 0.0 && x < 0)) { // A constant-evaluation context: Use a constexpr-friendly algorithm. 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 { // Let the code generator figure it out. return std::pow(b, double(x)); } } int main() { // A constant-expression context constexpr double kilo = power(10.0, 3); int n = 3; // Not a constant expression, because n cannot be converted to an rvalue // in a constant-expression context // Equivalent to std::pow(10.0, double(n)) double mucho = power(10.0, n); std::cout << kilo << " " << mucho << "\n"; // (3) }
Ausgabe
1000 1000
[bearbeiten] Siehe auch
constexpr-Spezifizierer(C++11) |
gibt an, dass der Wert einer Variablen oder Funktion zur Kompilierzeit berechnet werden kann |
consteval-Spezifizierer(C++20) |
gibt an, dass eine Funktion eine unmittelbare Funktion ist, d. h., jeder Aufruf der Funktion muss in einer konstanten Auswertung erfolgen |
constinit-Spezifizierer(C++20) |
stellt sicher, dass eine Variable eine statische Initialisierung hat, d. h. Null-Initialisierung und konstante Initialisierung |