美文网首页
c++11 移动语义

c++11 移动语义

作者: 胖子罗 | 来源:发表于2020-03-29 10:58 被阅读0次

测试代码

#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.png
image.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、移动的作用在于避免拷贝提高效率,并转移使用权。

相关文章

网友评论

      本文标题:c++11 移动语义

      本文链接:https://www.haomeiwen.com/subject/teuyuhtx.html