std::swappable, std::swappable_with

来自cppreference.com
< cpp‎ | concepts
定义于头文件 <concepts>
template< class T >

concept swappable =
  requires(T& a, T& b) {
    ranges::swap(a, b);

  };
(1) (C++20 起)
template< class T, class U >

concept swappable_with =
  common_reference_with<T, U> &&
  requires(T&& t, U&& u) {
    ranges::swap(std::forward<T>(t), std::forward<T>(t));
    ranges::swap(std::forward<U>(u), std::forward<U>(u));
    ranges::swap(std::forward<T>(t), std::forward<U>(u));
    ranges::swap(std::forward<U>(u), std::forward<T>(t));

  };
(2) (C++20 起)

概念 swappable<T> 指定 T 类型左值可交换。

概念 swappable_with<T, U> 指定属于 TU 所编码的类型和值类别的表达式彼此可交换。 swappable_with<T, U> 仅若调用 ranges::swap(t, u) 交换 tu 的值,即给定等于 t 的独立对象 t2 和等于 u 的独立对象 u2 ,求值 ranges::swap(t, u)ranges::swap(u, t)t2 等于 uu2 等于 t ,才得到满足。

相等性保持

若表达式对给定的相等输入产生相等输出,则它保持相等性

  • 表达式的输入由其操作数组成。
  • 表达式的输出由其结果和表达式所修改的所有操作数(若存在)组成。

在标准概念的规范中,操作数定义为仅包含下列内容的最大子表达式:

每个操作数的 cv 限定与值类别,是通过假设每个模板类型形参代表一个 cv 无限定的非数组对象类型确定的。

进一步要求每个要求保持相等性的表达式都稳定:这种表达式带相同输入对象的二次求值必须拥有相等的输出,而无任何对这些输入对象的显式中间修改。

除非另外提醒,每个用于 requires 表达式中的表达式都要求保持相等性且稳定,而表达式的求值必须只修改其非常操作数。必须不修改常操作数。