std::ranges::swap
来自 cppreference.com
定义在头文件 <concepts> 中 |
||
namespace ranges { inline namespace /* 未指定 */ { |
(自 C++20 起) (自定义点对象) |
|
调用签名 |
||
template< class T, class U > constexpr void ranges::swap( T&& t, U&& u ) noexcept(/* 见下文 */); |
(自 C++20 起) | |
交换 t 和 u 所引用的值。
ranges::swap(t, u) 与 表达式等价于
- (void)swap(t, u),如果 t 或 u 是类或枚举类型,并且该表达式有效,其中 重载解析是在命名空间
std::ranges
中执行的,并带有附加的候选 template<class T> void swap(T&, T&) = delete;.- 如果重载解析选择的函数没有交换 t 和 u 所引用的值,则程序格式错误;不需要诊断。
- 否则,(void)ranges::swap_ranges(t, u),如果 t 和 u 是相同范围的左值数组(但可能具有不同的元素类型),并且 ranges::swap(*t, *u) 是有效的表达式,除了 noexcept((void)ranges::swap_ranges(t, u)) 等于 noexcept(ranges::swap(*t, *u)).
- 否则,一个交换 t 和 u 的引用值的表达式,如果它们都是相同类型
V
的左值,并且该类型符合 std::move_constructible<V> 和 std::assignable_from<V&, V>。- 将
noexcept
运算符 应用于该表达式的结果等于 std::is_nothrow_move_constructible_v<V> && std::is_nothrow_move_assignable_v<V>. - 如果该表达式是一个 常量表达式,则
-
V
是一个 LiteralType, t = std::move(u))
和u = std::move(t)
都是 常量子表达式,并且- 以下声明中初始值设定项的 完整表达式 是常量子表达式
- V v1(std::move(t));
- V v2(std::move(u));
-
- 将
- 否则,ranges::swap(t, u) 格式错误,这会导致 替换失败,当 ranges::swap(t, u) 出现在模板实例化的直接上下文中时。
自定义点对象
名称 ranges::swap
表示一个自定义点对象,它是一个常量 函数对象,属于 字面量 semiregular
类类型。为了便于说明,其类型的 cv 无限定版本表示为 __swap_fn
。
所有 __swap_fn
实例都是相等的。调用不同类型 __swap_fn
实例对相同参数的影响是等效的,无论表示该实例的表达式是左值还是右值,并且是否有 const 限定(但是,不需要调用 volatile 限定实例)。因此,ranges::swap
可以自由复制,并且其副本可以互换使用。
对于一组类型 Args...
,如果 std::declval<Args>()... 满足上述 ranges::swap
参数的要求,则 __swap_fn
符合
- std::invocable<__swap_fn, Args...>,
- std::invocable<const __swap_fn, Args...>,
- std::invocable<__swap_fn&, Args...>, 并且
- std::invocable<const __swap_fn&, Args...>.
否则,__swap_fn
的任何函数调用运算符都不会参与重载解析。
[编辑] 示例
运行此代码
#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); }
输出
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
[编辑] 另请参阅
(C++20) |
指定类型是否可以互换,或者两个类型是否可以相互互换 (概念) |
交换两个对象的的值 (函数模板) |