std::unwrap_reference, std::unwrap_ref_decay
Von cppreference.com
< cpp | utility | functional
| Definiert in der Kopfdatei <type_traits> |
||
| Definiert in der Header-Datei <functional> |
||
| template< class T > struct unwrap_reference; |
(1) | (seit C++20) |
| template< class T > struct unwrap_ref_decay; |
(2) | (seit C++20) |
Entpackt jede std::reference_wrapper: ändert std::reference_wrapper<U> zu U&.
1) Wenn
T eine Spezialisierung von std::reference_wrapper ist, wird diese entpackt; andernfalls bleibt T unverändert.2) Wenn das zerlegte
T eine Spezialisierung von std::reference_wrapper ist, wird diese entpackt; andernfalls wird T zerlegt.Wenn das Programm Spezialisierungen für eine der auf dieser Seite beschriebenen Vorlagen hinzufügt, ist das Verhalten undefiniert.
Inhalt |
[bearbeiten] Verschachtelte Typen
| Typ | Definition |
type
|
(1) |
[bearbeiten] Hilfstypen
| template<class T> using unwrap_reference_t = unwrap_reference<T>::type; |
(1) | (seit C++20) |
| template<class T> using unwrap_ref_decay_t = unwrap_ref_decay<T>::type; |
(2) | (seit C++20) |
[bearbeiten] Mögliche Implementierung
template<class T> struct unwrap_reference { using type = T; }; template<class U> struct unwrap_reference<std::reference_wrapper<U>> { using type = U&; }; template<class T> struct unwrap_ref_decay : std::unwrap_reference<std::decay_t<T>> {}; |
[bearbeiten] Hinweise
std::unwrap_ref_decay führt die gleiche Transformation durch wie sie von std::make_pair und std::make_tuple verwendet wird.
| Feature-Test-Makro | Wert | Std | Feature |
|---|---|---|---|
__cpp_lib_unwrap_ref |
201811L |
(C++20) | std::unwrap_ref_decay und std::unwrap_reference |
[bearbeiten] Beispiel
Führen Sie diesen Code aus
#include <cassert> #include <functional> #include <iostream> #include <type_traits> int main() { static_assert(std::is_same_v<std::unwrap_reference_t<int>, int>); static_assert(std::is_same_v<std::unwrap_reference_t<const int>, const int>); static_assert(std::is_same_v<std::unwrap_reference_t<int&>, int&>); static_assert(std::is_same_v<std::unwrap_reference_t<int&&>, int&&>); static_assert(std::is_same_v<std::unwrap_reference_t<int*>, int*>); { using T = std::reference_wrapper<int>; using X = std::unwrap_reference_t<T>; static_assert(std::is_same_v<X, int&>); } { using T = std::reference_wrapper<int&>; using X = std::unwrap_reference_t<T>; static_assert(std::is_same_v<X, int&>); } static_assert(std::is_same_v<std::unwrap_ref_decay_t<int>, int>); static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int>, int>); static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int&>, int>); { using T = std::reference_wrapper<int&&>; using X = std::unwrap_ref_decay_t<T>; static_assert(std::is_same_v<X, int&>); } { auto reset = []<typename T>(T&& z) { // x = 0; // Error: does not work if T is reference_wrapper<> // converts T&& into T& for ordinary types // converts T&& into U& for reference_wrapper<U> decltype(auto) r = std::unwrap_reference_t<T>(z); std::cout << "r: " << r << '\n'; r = 0; // OK, r has reference type }; int x = 1; reset(x); assert(x == 0); int y = 2; reset(std::ref(y)); assert(y == 0); } }
Ausgabe
r: 1 r: 2
[bearbeiten] Siehe auch
| (C++11) |
CopyConstructible und CopyAssignable Referenz-Wrapper(Klassen-Template) |
erstellt ein pair-Objekt vom Typ, bestimmt durch die Argumenttypen(Funktionsvorlage) | |
| (C++11) |
erstellt ein tuple-Objekt vom Typ, der durch die Argumenttypen definiert ist(function template) |