为什么std::swap可以正确交换自定义对象
Why can std::swap correctly swap custom objects?
我编写了以下代码作为c++中正确实现swap
函数的示例。
#include <iostream>
class Complex
{
int a;
int b;
friend std::ostream& operator<<(std::ostream&, const Complex&);
friend void swap(Complex& rhs, Complex& lhs) noexcept;
public:
Complex(): a(0), b(0) {};
Complex(int a, int b): a(a), b(b) {};
};
void swap(Complex& rhs, Complex& lhs) noexcept
{
using std::swap;
swap(rhs.a, lhs.a);
swap(rhs.b, lhs.b);
}
std::ostream& operator<<(std::ostream& os, const Complex& c)
{
os << c.a << c.b;
return os;
}
class Swapable
{
int a;
int b;
Complex complex;
friend std::ostream& operator<<(std::ostream&, const Swapable&);
friend void swap(Swapable&, Swapable&) noexcept;
public:
Swapable(): a(0), b(0) {};
Swapable(int a, int b): a(a), b(b), complex(a+1, b+1) {};
};
void swap(Swapable& rhs, Swapable& lhs) noexcept
{
std::cout << "Swapping" << std::endl;
using std::swap;
swap(rhs.a, lhs.a);
swap(rhs.b, lhs.b);
swap(rhs.complex, lhs.complex);
}
std::ostream& operator<<(std::ostream& os, const Swapable& s)
{
os << s.a << s.b << s.complex;
return os;
}
int main()
{
Swapable s_1(1,1);
Swapable s_2(2,2);
std::cout << "Before swap" << std::endl;
std::cout << s_1 << std::endl;
std::cout << s_2 << std::endl;
swap(s_1, s_2);
std::cout << "After swap" << std::endl;
std::cout << s_1 << std::endl;
std::cout << s_2 << std::endl;
std::swap(s_1, s_2); // It should fail. Shouldn't it?
std::cout << "Second swap" << std::endl;
std::cout << s_1 << std::endl;
std::cout << s_2 << std::endl;
}
输出为:
Before swap
1122
2233
Swapping
After swap
2233
1122
Second swap
1122
2233
当我调用不合格的swap
时,一切都如预期的那样工作。然而,当我用自定义对象调用std::swap
时,我预计会出现编译错误。为什么std::swap
函数能够正确地交换我的自定义对象?
类Swapable
是MoveAssignable和MoveConstructible,那么std::swap可以很好地工作。
类型需求
T
必须满足MoveAssignable和MoveConstructible的要求。
这意味着std::map
可以使用Swapable
提供的move-assign和move-construct操作来完成这些工作。类Swapable
满足要求,它有隐式声明的移动构造函数和移动赋值操作符,(以及隐式声明的复制构造函数和复制赋值操作符)
相关文章:
- C++映射:具有自定义类的运算符[]不起作用(总是返回0)
- 如何将点击的信号和插槽添加到qt中的自定义按钮中
- C++自定义比较函数
- 如何比较自定义类的std::变体
- std::设置自定义比较器
- 如何正确实现和访问运算符的各种自定义枚举器
- flutter:即使shouldRepaint()返回true,自定义画家也不会重新绘制
- 自定义先决条件对移动分配运算符有效吗
- 使用VS Code和CMake Tools运行自定义命令
- 如何创建从Maya(或类似程序)到虚幻引擎的自定义数据导出插件
- std::ranges::elements_view,用于自定义类似元组的数据
- 跟随整数索引列表的自定义类迭代器
- 参数化自定义CMake工具链
- 使用自定义比较函数使用std::sort()对矢量字符串进行排序时出现问题
- C++ - 没有自定义交换功能的移动分配运算符?
- std::排序和自定义交换函数
- 在 std::sort() 中的自定义函数中,>(大于)运算符与 <(小于运算符)的交换会导致分段错误吗?
- 为什么std::swap可以正确交换自定义对象
- 移动语义==自定义交换函数已过时
- 带有自定义交换的c++排序实现