std::ranges::swap

来自cppreference.com
< cpp‎ | utility
 
 
工具库
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)
(C++20)
swap 与类型运算
ranges::swap
(C++20)
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)
 
在标头 <concepts> 定义
inline namespace /* unspecified */ {

    inline constexpr /* unspecified */ swap = /* unspecified */;

}
(C++20 起)
(定制点对象)
调用签名
template< class T, class U >

    requires /* see below */

constexpr void swap(T&& t, U&& u) noexcept(/* see below */);

交换 tu 所引用的值。

调用 ranges::swap

  1. 表达式等价于 (void)swap(std::forward<T>(t), std::forward<U>(u)) ,若以下列候选进行重载决议则该表达式合法:
    • template<class T> void swap(T&, T&) = delete;
    • template<class T, std::size_t N> void swap(T(&)[N], T(&)[N]) = delete;
    • 实参依赖查找所找到的任何 swap 的声明。
    若重载决议所选择的函数不交换 tu 所引用的值,则程序为谬构;不要求诊断。
  2. 否则表达式等价于 (void)ranges::swap_ranges(t, u) ,除了 noexcept(ranges::swap(t), ranges::swap(u)) 等于 noexcept(ranges::swap(*t, *u)) ,若 TU 为到相等长度(但元素类型可能不同)的数组类型的左值引用,且 ranges::swap(*t, *u) 为合法表达式;
  3. 否则如同用 V v(std::move(t)); t = std::move(u); u = std::move(v); 交换其被引用值,若 TU 均为 V& ,而 V 是满足 std::move_constructible<V>std::assignable_from<V&, V> 的某个类型,
    • 调用是否为潜在抛出由上述操作指定。
    • 若上述操作与以交换 uv 进行的上述操作均可用于常量求值,则调用为常量子表达式。
    若任一概念未被实现,则程序为谬构;不要求诊断。
  4. 所有其他情况下,对 ranges::swap 的调用为谬构,这能在 ranges::swap(t, u) 出现于模板实例化的立即语境时导致替换失败

表达式等价

表达式 e 表达式等价于表达式 f ,若 ef 拥有相同效果,均为潜在抛出或均非潜在抛出(即 noexcept(e) == noexcept(f) ),且均为常量子表达式或均非常量子表达式。

定制点对象

名字 ranges::swap 代表一个定制点对象,它是字面 semiregular 类类型的 const 函数对象。为说明目的,以 __swap_fn 表示其类型的 cv 无限定版本。

__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 const 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 const name, int p, int q)
{
    std::cout << name << "1 = " << p << ", " << name << "2 = " << q << '\n';
}
 
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); // 错误:类型不匹配
 
    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); // 错误:类型不匹配
 
    // char f1[]{1,2,3};
    // int  f2[]{4,5,6};
    // std::ranges::swap(f1, f2); // 错误:类型不匹配
 
    int g1{27}, g2{28};
    std::ranges::swap(g1, g2);
    print("g", g1, g2);
}

输出:

a1{ 13 14 }, a2{ 10 11 12 }
b1{ 18 19 20 }, b2{ 15 16 17 }
d1{ 24 25 26 }, d2{ 21 22 23 }
g1 = 28, g2 = 27

参阅

指定一个类型能进行交换,或两个类型能彼此交换
(概念)
交换两个对象的值
(函数模板)