Namensräume
Varianten
Aktionen

std::is_constant_evaluated

Von cppreference.com
< cpp‎ | types
 
 
Dienstprogramm-Bibliotheken
Sprachunterstützung
Typunterstützung (Basistypen, RTTI)
Bibliotheks-Feature-Test-Makros (C++20)
Programm-Dienstprogramme
Variadische Funktionen
is_constant_evaluated
(C++20)
Coroutine-Unterstützung (C++20)
Vertragsunterstützung (C++26)
Drei-Wege-Vergleich
(C++20)
(C++20)(C++20)(C++20)  
(C++20)(C++20)(C++20)

Allgemeine Hilfsmittel
Relationale Operatoren (in C++20 veraltet)
 
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[edit]
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[bearbeiten]
constinit-Spezifizierer(C++20) stellt sicher, dass eine Variable eine statische Initialisierung hat, d. h. Null-Initialisierung und konstante Initialisierung[bearbeiten]