测试代码
#include <iostream>
using namespace std;
class Person
{
private:
int age;
string name;
int* data;
public:
Person() : data(new int[1000000]){}
~Person() { delete [] data; }
// 拷贝构造函数
Person(const Person& p) :
age(p.age),
name(p.name),
data(new int[1000000]){
std::copy(p.data, p.data+1000000, data);
cout << "Copy Constructor" << endl;
}
// 拷贝赋值运算符
Person& operator=(const Person& p){
this->age = p.age;
this->name = p.name;
this->data = new int[1000000];
std::copy(p.data, p.data+1000000, data);
cout << "Copy Assign" << endl;
return *this;
}
// 移动构造函数
Person(Person &&p) :
age(std::move(p.age)),
name(std::move(p.name)),
data(p.data){
p.data=nullptr; // 源对象的指针应该置空,以免源对象析构时影响本对象
cout << "Move Constructor" << endl;
}
// 移动赋值运算符
Person& operator=(Person &&p){
this->age = std::move(p.age);
this->name = std::move(p.name);
this->data = p.data;
p.data=nullptr;
cout << "Move Assign" << endl;
return *this;
}
// 普通成员函数
void PrintAge(){
cout << "age:" << this->age << endl;
cout << "data:" << this->data << endl;
}
};
int main(){
Person p1;
Person p2 = p1; // 拷贝构造函数
p1.PrintAge();
p2.PrintAge();
Person p3,p4;
p3 = p4; // 拷贝赋值运算符
p3.PrintAge();
p4.PrintAge();
Person p5;
Person p6 = std::move(p5); // 移动构造函数
p5.PrintAge();
p6.PrintAge();
Person p7,p8;
p7 = std::move(p8); // 移动赋值运算符
p7.PrintAge();
p8.PrintAge();
return 0;
}
实际应用代码:
CodePointMatcherWarehouse.pngimage.png
这是node源码的一个CodePointMatcherWarehouse类,从上图的声明和定义看到这个类只有移动拷贝构造函数和移动赋值运算符,而没有拷贝构造函数和拷贝赋值运算符。
还有一个特殊的宏: U_NOEXCEPT
#ifdef U_NOEXCEPT
/* Use the predefined value. */
#else
# define U_NOEXCEPT noexcept
#endif
这就要求尽量保证移动构造函数、移动拷贝赋值运算符不发生异常;使用noexcept关键字,是为了保证移动构造函数中抛出来的异常会直接调用terminate终止程序。
总结
1、使用&&定义移动拷贝构造和移动赋值运算符;
2、调用时使用std::move函数;
3、移动的作用在于避免拷贝提高效率,并转移使用权。
网友评论