std::ranges::swap
| Definiert in der Header-Datei <concepts> |
||
| namespace ranges { inline namespace /* nicht spezifiziert */ { |
(seit C++20) (Customization-Point-Objekt) |
|
| Aufruf-Signatur |
||
| template< class T, class U > constexpr void ranges::swap( T&& t, U&& u ) noexcept(/* siehe unten */); |
(seit C++20) | |
Tauscht die Werte, auf die durch t und u verwiesen wird.
ranges::swap(t, u) ist ausdrucksäquivalent zu
- (void)swap(t, u), wenn t oder u einen Klassen- oder Aufzählungstyp haben, und dieser Ausdruck gültig ist, wobei die Überladungsauswahl im Namespace
std::rangesmit dem zusätzlichen Kandidaten template<class T> void swap(T&, T&) = delete; durchgeführt wird.- Wenn die durch die Überladungsauswahl ausgewählte Funktion die Werte, auf die durch t und u verwiesen wird, nicht vertauscht, ist das Programm schlecht geformt; keine Diagnose erforderlich.
- Andernfalls (void)ranges::swap_ranges(t, u), wenn t und u L-Wert-Arrays gleicher Ausdehnung (aber möglicherweise unterschiedlicher Elementtypen) sind und ranges::swap(*t, *u) ein gültiger Ausdruck ist, wobei noexcept((void)ranges::swap_ranges(t, u)) gleich noexcept(ranges::swap(*t, *u)) ist.
- Andernfalls ein Ausdruck, der die referenzierten Werte von t und u vertauscht, wenn beide L-Werte des gleichen Typs
Vsind, der std::move_constructible<V> und std::assignable_from<V&, V> modelliert.- Das Ergebnis der Anwendung des
noexcept-Operators auf diesen Ausdruck ist gleich std::is_nothrow_move_constructible_v<V> && std::is_nothrow_move_assignable_v<V>. - Dieser Ausdruck ist ein Konstantausdruck, wenn
-
Vein LiteralType ist, - beide t = std::move(u)) und u = std::move(t) Konstante Unterausdrücke sind, und
- die vollständigen Ausdrücke der Initialisierer in den folgenden Deklarationen Konstante Unterausdrücke sind
- V v1(std::move(t));
- V v2(std::move(u));
-
- Das Ergebnis der Anwendung des
- Andernfalls ist ranges::swap(t, u) schlecht geformt, was zu einem Substitutionsfehler führen kann, wenn ranges::swap(t, u) im unmittelbaren Kontext einer Template-Instanziierung auftritt.
Customization Point Objects
Der Name ranges::swap bezeichnet ein Customization Point Object, das ein const Funktionsobjekt eines literalen semiregular-Klassentyps ist. Zu Auslegungszwecken wird die cv-unqualifizierte Version seines Typs als __swap_fn bezeichnet.
Alle Instanzen von __swap_fn sind gleich. Die Auswirkungen des Aufrufs unterschiedlicher Instanzen vom Typ __swap_fn mit denselben Argumenten sind äquivalent, unabhängig davon, ob der Ausdruck, der die Instanz bezeichnet, ein lvalue oder rvalue ist, und ob er const-qualifiziert ist oder nicht (jedoch ist keine volatile-qualifizierte Instanz aufrufbar).
Gegeben sei eine Menge von Typen Args.... Wenn std::declval<Args>()... die Anforderungen für Argumente an ranges::swap oben erfüllt, modelliert __swap_fn
- std::invocable<__swap_fn, Args...>,
- std::invocable<const __swap_fn, Args...>,
- std::invocable<__swap_fn&, Args...>, und
- std::invocable<const __swap_fn&, Args...>.
Andernfalls nimmt kein Funktionsaufrufoperator von __swap_fn an der Überladungsauswahl teil.
[edit] Beispiel
#include <array> #include <concepts> #include <iostream> #include <ranges> #include <string_view> #include <vector> void print(std::string_view name, std::ranges::common_range auto const& p, std::ranges::common_range auto const& q) { std::cout << name << "1{ "; for (auto const& i : p) std::cout << i << ' '; std::cout << "}, " << name << "2{ "; for (auto const& i : q) std::cout << i << ' '; std::cout << "}\n"; } void print(std::string_view name, int p, int q) { std::cout << name << "1 = " << p << ", " << name << "2 = " << q << '\n'; } struct IntLike { int v; }; void swap(IntLike& lhs, int& rhs) { std::swap(lhs.v, rhs); } void swap(int& lhs, IntLike& rhs) { std::swap(lhs, rhs.v); } std::ostream& operator<<(std::ostream& out, IntLike i) { return out << i.v; } int main() { std::vector a1{10, 11, 12}, a2{13, 14}; std::ranges::swap(a1, a2); print("a", a1, a2); std::array b1{15, 16, 17}, b2{18, 19, 20}; std::ranges::swap(b1, b2); print("b", b1, b2); // std::array c1{1, 2, 3}; std::array c2{4, 5}; // std::ranges::swap(c1, c2); // error: no swap found by ADL int d1[]{21, 22, 23}, d2[]{24, 25, 26}; std::ranges::swap(d1, d2); print("d", d1, d2); // int e1[]{1, 2, 3}, e2[]{4, 5}; // std::ranges::swap(e1, e2); // error: extents mismatch // char f1[]{1, 2, 3}; // int f2[]{4, 5, 6}; // std::ranges::swap(f1, f2); // error: no swap(*f1, *f2) found by ADL IntLike g1[]{1, 2, 3}; int g2[]{4, 5, 6}; std::ranges::swap(g1, g2); // heterogeneous swap supported print("g", g1, g2); int h1{27}, h2{28}; std::ranges::swap(h1, h2); print("h", h1, h2); }
Ausgabe
a1{ 13 14 }, a2{ 10 11 12 }
b1{ 18 19 20 }, b2{ 15 16 17 }
d1{ 24 25 26 }, d2{ 21 22 23 }
g1{ 4 5 6 }, g2{ 1 2 3 }
h1 = 28, h2 = 27[edit] Siehe auch
| (C++20) |
gibt an, dass ein Typ vertauscht werden kann oder dass zwei Typen miteinander vertauscht werden können (Konzept) |
| tauscht die Werte von zwei Objekten (Funktionsvorlage) |