Bootstrap

C++复制构造函数&移动构造函数,复制赋值运算符&移动赋值运算符

一、调用时机


1、复制构造函数调用的时机

·对象在创建时使用其他的对象初始化

Person p(q); //此时复制构造函数被用来创建实例p

Person p = q; //此时复制构造函数被用来在定义实例p时初始化p

return_p()  //当函数返回该类型的对象时调用


·对象作为函数的参数进行值传递时

f(p); //此时p作为函数的参数进行值传递,p入栈时会调用复制构造函数创建一个局部对象,与函数内的局部变量具有相同的作用域

p = q; //此时没有复制构造函数的调用!赋值并不会调用复制构造函数,赋值只是赋值运算符(重载)在起作用

简单来记的话就是,如果对象在声明的同时将另一个已存在的对象赋给它,就会调用复制构造函数;如果对象已经存在,然后将另一个已存在的对象赋给它,调用的就是赋值运算符(重载)

注意:默认的复制构造函数和赋值运算符进行的都是"shallow copy",浅复制——在C++中,在用一个对象初始化另一个对象时,只复制了成员,并没有复制资源,使两个对象同时指向了同一资源的复制方式称为浅复制。浅复制存在资源矛盾,导致两次调用析构函数时后面调用者出错。
因此如果对象中含有动态分配的内存,就需要我们自己重写复制构造函数或者重载赋值运算符来实现"deep copy",确保数据的完整性和安全性。

2、移动构造函数调用的时机

首先有移动构造函数并且可以访问。

移动构造函数接收的是“右值引用”的参数。
移动构造函数何时触发?  那就是临时对象(右值)。用到临时对象的时候就会执行移动语义。
T &&a = return_a();  如果临时对象即将消亡,并且它里面的资源是需要被其他对象再引用用的,这个时候我们就可以触发移动构造。
T b = return_b(); 复制构造

此时a是右值引用,他比b少了一次对象构造和对象析构的过程。a直接绑定了return_a()返回的临时变量。
而b只是由临时变量值(值拷贝)构造而成的。
应该可以看清楚了吧。右值引用就是让返回的右值(临时对象)重获新生,延长生命周期。临时对象析构了,但是右值引用存活。


不过要注意的是,右值引用不能绑定左值:
int a;
int &&c = a;   这样是不行的。


3、复制赋值运算符


4、移动赋值运算符






















;