第13章 拷贝控制
一个类通过定义五种特殊的成员函数来控制对象的拷贝、移动、赋值和销毁操作。
-
拷贝构造函数(copy constructor)
-
拷贝赋值运算符(copy-assignment operator)
-
移动构造函数(move constructor)
-
移动赋值运算符(move-assignment operator)
-
析构函数(destructor)
这些操作统称为拷贝控制操作(copy control)。
在定义任何类时,拷贝控制操作都是必要部分。
拷贝、赋值与销毁(Copy,Assign,and Destroy)
拷贝构造函数(The Copy Constructor)
如果一个构造函数的第一个参数是自身类类型的引用(几乎总是const
引用),且任何额外参数都有默认值,则此构造函数是拷贝构造函数。
由于拷贝构造函数在一些情况下会被隐式使用,因此通常不会声明为explicit
的。
如果类未定义自己的拷贝构造函数,编译器会为类合成一个。一般情况下,合成拷贝构造函数(synthesized copy constructor)会将其参数的非static
成员逐个拷贝到正在创建的对象中。
class Sales_data
{
public:
// other members and constructors as before
// declaration equivalent to the synthesized copy constructor
Sales_data(const Sales_data&);
private:
std::string bookNo;
int units_sold = 0;
double revenue = 0.0;
};
// equivalent to the copy constructor that would be synthesized for Sales_data
Sales_data::Sales_data(const Sales_data &orig):
bookNo(orig.bookNo), // uses the string copy constructor
units_sold(orig.units_sold), // copies orig.units_sold
revenue(orig.revenue) // copies orig.revenue
{ } // empty bod
使用直接初始化时,实际上是要求编译器按照函数匹配规则来选择与实参最匹配的构造函数。使用拷贝初始化时,是要求编译器将右侧运算对象拷贝到正在创建的对象中,如果需要的话还要进行类型转换。
string dots(10, '.'); // direct initialization
string s(dots); // direct initialization
string s2 = dots; // copy initialization
string null_book = "9-999-99999-9"; // copy initialization
string nines = string(100, '9'); // copy initialization
拷贝初始化通常使用拷贝构造函数来完成。但如果一个类拥有移动构造函数,则拷贝初始化有时会使用移动构造函数而非拷贝构造函数来完成。
发生拷贝初始化的情况:
-
用
=
定义变量。 -
将对象作为实参传递给非引用类型的形参。
-
从返回类型为非引用类型的函数返回对象。
-
用花括号列表初始化数组中的元素或聚合类中的成员。
当传递一个实参或者从函数返回一个值时,不能隐式使用explicit
构造函数。
vector<int> v1(10); // ok: direct initialization
vector<int> v2 = 10; // error: constructor that takes a size is explicit
void f(vector<int>); // f's parameter is copy initialized
f(10); // error: can't use an explicit constructor to copy an argument
f(vector<int>(10)); // ok: directly construct a temporary vector from an int
拷贝赋值运算符(The Copy-Assignment Operator)
重载运算符(overloaded operator)的参数表示运算符的运算对象。
如果一个运算符是成员函数,则其左侧运算对象会绑定到隐式的this
参数上。
赋值运算符通常应该返回一个指向其左侧运算对象的引用。
标准库通常要求保存在容器中的类型要具有赋值运算符,且其返回值是左侧运算对象的引用。
如果类未定义自己的拷贝赋值运算符,编译器会为类合成一个。一般情况下,合成拷贝赋值运算符(synthesized copy-assignment operator)会将其右侧运算对象的非static
成员逐个赋值给左侧运算对象的对应成员,之后返回左侧运算对象的引用。
// equivalent to the synthesized copy-assignment operator
Sales_data& Sales_data::operator=(const Sales_data &rhs)
{
bookNo = rhs.bookNo; // calls the string::operator=
units_sold = rhs.units_sold; // uses the built-in int assignment
revenue = rhs.revenue; // uses the built-in double assignment
return *this; // return a reference to this object
}
析构函数(The Destructor)
析构函数负责释放对象使用的资源,并销毁对象的非static
数据成员。
析构函数的名字由波浪号~
接类名构成,它没有返回值,也不接受参数。
由于析构函数不接受参数,所以它不能被重载。
如果类未定义自己的析构函数,编译器会为类合成一个。合成析构函数(synthesized destructor)的函数体为空。
析构函数首先执行函数体,然后再销毁数据成员。在整个对象销毁过程中,析构函数体是作为成员销毁步骤之外的另一部分而进行的。成员按照初始化顺序的逆序销毁。
隐式销毁一个内置指针类型的成员不会delete
它所指向的对象。
无论何时一个对象被销毁,都会自动调用其析构函数。
当指向一个对象的引用或指针离开作用域时,该对象的析构函数不会执行。
三/五法则(The Rule of Three/Five)
需要析构函数的类一般也需要拷贝和赋值操作。
需要拷贝操作的类一般也需要赋值操作,反之亦然。
使用=default
(Using =default
)
可以通过将拷贝控制成员定义为=default
来显式地要求编译器生成合成版本。
class Sales_data
{
public:
// copy control; use defaults
Sales_data() = default;
Sales_data(const Sales_data&) = default;
~Sales_data() = default;
// other members as before
};
在类内使用=default
修饰成员声明时,合成的函数是隐式内联的。如果不希望合成的是内联函数,应该只对成员的类外定义使用=default
。
只能对具有合成版本的成员函数使用=default
。
阻止拷贝(Preventing Copies)
大多数类应该定义默认构造函数、拷贝构造函数和拷贝赋值运算符,无论是显式地还是隐式地。
在C++11新标准中,将拷贝构造函数和拷贝赋值运算符定义为删除的函数(deleted function)可以阻止类对象的拷贝。删除的函数是一种虽然进行了声明,但是却不能以任何方式使用的函数。定义删除函数的方式是在函数的形参列表后面添加=delete
。
struct NoCopy
{
NoCopy() = default; // use the synthesized default constructor
NoCopy(const NoCopy&) = delete; // no copy
NoCopy &operator=(const NoCopy&) = delete; // no assignment
~NoCopy() = default; // use the synthesized destructor
// other members
};
=delete
和=default
有两点不同:
-
=delete
可以对任何函数使用;=default
只能对具有合成版本的函数使用。 -
=delete
必须出现在函数第一次声明的地方;=default
既能出现在类内,也能出现在类外。
析构函数不能是删除的函数。对于析构函数被删除的类型,不能定义该类型的变量或者释放指向该类型动态分配对象的指针。
如果一个类中有数据成员不能默认构造、拷贝或销毁,则对应的合成拷贝控制成员将被定义为删除的。
在旧版本的C++标准中,类通过将拷贝构造函数和拷贝赋值运算符声明为private
成员来阻止类对象的拷贝。在新标准中建议使用=delete
而非private
。
拷贝控制和资源管理(Copy Control and Resource Management)
通常,管理类外资源的类必须定义拷贝控制成员。
行为像值的类(Classes That Act Like Values)
class HasPtr
{
public:
HasPtr(const std::string &s = std::string()):
ps(new std::string(s)), i(0) { }
// each HasPtr has its own copy of the string to which ps points
HasPtr(const HasPtr &p):
ps(new std::string(*p.ps)), i(p.i) { }
HasPtr& operator=(const HasPtr &);
~HasPtr() { delete ps; }
private:
std::string *ps;
int i;
};
编写赋值运算符时有两点需要注意:
- 即使将一个对象赋予它自身,赋值运算符也能正确工作。
// WRONG way to write an assignment operator!
HasPtr& HasPtr::operator=(const HasPtr &rhs)
{
delete ps; // frees the string to which this object points
// if rhs and *this are the same object, we're copying from deleted memory!
ps = new string(*(rhs.ps));
i = rhs.i;
return *this;
}
- 赋值运算符通常结合了拷贝构造函数和析构函数的工作。
编写赋值运算符时,一个好的方法是先将右侧运算对象拷贝到一个局部临时对象中。拷贝完成后,就可以安全地销毁左侧运算对象的现有成员了。
HasPtr& HasPtr::operator=(const HasPtr &rhs)
{
auto newp = new string(*rhs.ps); // copy the underlying string
delete ps; // free the old memory
ps = newp; // copy data from rhs into this object
i = rhs.i;
return *this; // return this object
}
定义行为像指针的类(Defining Classes That Act Like Pointers)
class HasPtr
{
public:
// constructor allocates a new string and a new counter, which it sets to 1
HasPtr(const std::string &s = std::string()):
ps(new std::string(s)), i(0), use(new std::size_t(1)) {}
// copy constructor copies all three data members and increments the counter
HasPtr(const HasPtr &p):
ps(p.ps), i(p.i), use(p.use) { ++*use; }
HasPtr& operator=(const HasPtr&);
~HasPtr();
private:
std::string *ps;
int i;
std::size_t *use; // member to keep track of how many objects share *ps
};
析构函数释放内存前应该判断是否还有其他对象指向这块内存。
HasPtr::~HasPtr()
{
if (--*use == 0)
{ // if the reference count goes to 0
delete ps; // delete the string
delete use; // and the counter
}
}
HasPtr& HasPtr::operator=(const HasPtr &rhs)
{
++*rhs.use; // increment the use count of the right-hand operand
if (--*use == 0)
{ // then decrement this object's counter
delete ps; // if no other users
delete use; // free this object's allocated members
}
ps = rhs.ps; // copy data from rhs into this object
i = rhs.i;
use = rhs.use;
return *this; // return this object
}
交换操作(Swap)
通常,管理类外资源的类会定义swap
函数。如果一个类定义了自己的swap
函数,算法将使用自定义版本,否则将使用标准库定义的swap
。
class HasPtr
{
friend void swap(HasPtr&, HasPtr&);
// other members as in § 13.2.1 (p. 511)
};
inline void swap(HasPtr &lhs, HasPtr &rhs)
{
using std::swap;
swap(lhs.ps, rhs.ps); // swap the pointers, not the string data
swap(lhs.i, rhs.i); // swap the int members
}
一些算法在交换两个元素时会调用swap
函数,其中每个swap
调用都应该是未加限定的。如果存在类型特定的swap
版本,其匹配程度会优于std
中定义的版本(假定作用域中有using
声明)。
void swap(Foo &lhs, Foo &rhs)
{
// WRONG: this function uses the library version of swap, not the HasPtr version
std::swap(lhs.h, rhs.h);
// swap other members of type Foo
}
void swap(Foo &lhs, Foo &rhs)
{
using std::swap;
swap(lhs.h, rhs.h); // uses the HasPtr version of swap
// swap other members of type Foo
}
与拷贝控制成员不同,swap
函数并不是必要的。但是对于分配了资源的类,定义swap
可能是一种重要的优化手段。
由于swap
函数的存在就是为了优化代码,所以一般将其声明为内联函数。
定义了swap
的类通常用swap
来实现赋值运算符。在这种版本的赋值运算符中,右侧运算对象以值方式传递,然后将左侧运算对象与右侧运算对象的副本进行交换(拷贝并交换,copy and swap)。这种方式可以正确处理自赋值情况。
// note rhs is passed by value, which means the HasPtr copy constructor
// copies the string in the right-hand operand into rhs
HasPtr& HasPtr::operator=(HasPtr rhs)
{
// swap the contents of the left-hand operand with the local variable rhs
swap(*this, rhs); // rhs now points to the memory this object had used
return *this; // rhs is destroyed, which deletes the pointer in rhs
}
拷贝控制示例(A Copy-Control Example)
拷贝赋值运算符通常结合了拷贝构造函数和析构函数的工作。在这种情况下,公共部分应该放在private
的工具函数中完成。
动态内存管理类(Classes That Manage Dynamic Memory)
移动构造函数通常是将资源从给定对象“移动”而不是拷贝到正在创建的对象中。
对象移动(Moving Objects)
某些情况下,一个对象拷贝后就立即被销毁了,此时移动而非拷贝对象会大幅度提高性能。
在旧版本的标准库中,容器所能保存的类型必须是可拷贝的。但在新标准中,可以用容器保存不可拷贝,但可移动的类型。
标准库容器、string
和shared_ptr
类既支持移动也支持拷贝。IO类和unique_ptr
类可以移动但不能拷贝。
右值引用(Rvalue Reference)
为了支持移动操作,C++11引入了右值引用类型。右值引用就是必须绑定到右值的引用。可以通过&&
来获得右值引用。
int i = 42;
int &r = i; // ok: r refers to i
int &&rr = i; // error: cannot bind an rvalue reference to an
int &r2 = i * 42; // error: i * 42 is an rvalue
const int &r3 = i * 42; // ok: we can bind a reference to const to an rvalue
int &&rr2 = i * 42; // ok: bind rr2 to the result of the multiplication
右值引用只能绑定到即将被销毁,并且没有其他用户的临时对象上。使用右值引用的代码可以自由地接管所引用对象的资源。
变量表达式都是左值,所以不能将一个右值引用直接绑定到一个变量上,即使这个变量的类型是右值引用也不行。
int &&rr1 = 42; // ok: literals are rvalues
int &&rr2 = rr1; // error: the expression rr1 is an lvalue!
调用move
函数可以获得绑定在左值上的右值引用,此函数定义在头文件utility
中。
调用move
函数的代码应该使用std::move
而非move
,这样做可以避免潜在的名字冲突。
移动构造函数和移动赋值运算符(Move Constructor and Move Assignment)
移动构造函数的第一个参数是该类类型的右值引用,其他任何额外参数都必须有默认值。
除了完成资源移动,移动构造函数还必须确保移后源对象是可以安全销毁的。
在函数的形参列表后面添加关键字noexcept
可以指明该函数不会抛出任何异常。
对于构造函数,noexcept
位于形参列表和初始化列表开头的冒号之间。在类的头文件声明和定义中(如果定义在类外)都应该指定noexcept
。
class StrVec
{
public:
StrVec(StrVec&&) noexcept; // move constructor
// other members as before
};
StrVec::StrVec(StrVec &&s) noexcept : /* member initializers */
{ /* constructor body */ }
标准库容器能对异常发生时其自身的行为提供保障。虽然移动操作通常不抛出异常,但抛出异常也是允许的。为了安全起见,除非容器确定元素类型的移动操作不会抛出异常,否则在重新分配内存的过程中,它就必须使用拷贝而非移动操作。
不抛出异常的移动构造函数和移动赋值运算符必须标记为noexcept
。
在移动操作之后,移后源对象必须保持有效的、可销毁的状态,但是用户不能使用它的值。
StrVec &StrVec::operator=(StrVec &&rhs) noexcept
{
// direct test for self-assignment
if (this != &rhs)
{
free(); // free existing elements
elements = rhs.elements; // take over resources from rhs
first_free = rhs.first_free;
cap = rhs.cap;
// leave rhs in a destructible state
rhs.elements = rhs.first_free = rhs.cap = nullptr;
}
return *this;
}
只有当一个类没有定义任何拷贝控制成员,且类的每个非static
数据成员都可以移动时,编译器才会为类合成移动构造函数和移动赋值运算符。编译器可以移动内置类型的成员。如果一个成员是类类型,且该类有对应的移动操作,则编译器也能移动该成员。
// the compiler will synthesize the move operations for X and hasX
struct X
{
int i; // built-in types can be moved
std::string s; // string defines its own move operations
};
struct hasX
{
X mem; // X has synthesized move operations
};
X x, x2 = std::move(x); // uses the synthesized move constructor
hasX hx, hx2 = std::move(hx); // uses the synthesized move constructor
与拷贝操作不同,移动操作永远不会被隐式定义为删除的函数。但如果显式地要求编译器生成=default
的移动操作,且编译器不能移动全部成员,则移动操作会被定义为删除的函数。
定义了移动构造函数或移动赋值运算符的类必须也定义自己的拷贝操作,否则这些成员会被默认地定义为删除的函数。
如果一个类有可用的拷贝构造函数而没有移动构造函数,则其对象是通过拷贝构造函数来“移动”的,即使调用move
函数时也是如此。拷贝赋值运算符和移动赋值运算符的情况类似。
class Foo
{
public:
Foo() = default;
Foo(const Foo&); // copy constructor
// other members, but Foo does not define a move constructor
};
Foo x;
Foo y(x); // copy constructor; x is an lvalue
Foo z(std::move(x)); // copy constructor, because there is no move constructor
使用非引用参数的单一赋值运算符可以实现拷贝赋值和移动赋值两种功能。依赖于实参的类型,左值被拷贝,右值被移动。
// assignment operator is both the move- and copy-assignment operator
HasPtr& operator=(HasPtr rhs)
{
swap(*this, rhs);
return *this;
}
hp = hp2; // hp2 is an lvalue; copy constructor used to copy hp2
hp = std::move(hp2); // move constructor moves hp2
建议将五个拷贝控制成员当成一个整体来对待。如果一个类需要任何一个拷贝操作,它就应该定义所有五个操作。
移动赋值运算符可以直接检查自赋值情况。
C++11标准库定义了移动迭代器(move iterator)适配器。一个移动迭代器通过改变给定迭代器的解引用运算符的行为来适配此迭代器。移动迭代器的解引用运算符返回一个右值引用。
调用make_move_iterator
函数能将一个普通迭代器转换成移动迭代器。原迭代器的所有其他操作在移动迭代器中都照常工作。
最好不要在移动构造函数和移动赋值运算符这些类实现代码之外的地方随意使用move
操作。
右值引用和成员函数(Rvalue References and Member Functions)
区分移动和拷贝的重载函数通常有一个版本接受一个const T&
参数,另一个版本接受一个T&&
参数(T
为类型)。
void push_back(const X&); // copy: binds to any kind of X
void push_back(X&&); // move: binds only to modifiable rvalues of type X
有时可以对右值赋值:
在旧标准中,没有办法阻止这种使用方式。为了维持向下兼容性,新标准库仍然允许向右值赋值。但是可以在自己的类中阻止这种行为,规定左侧运算对象(即this
指向的对象)必须是一个左值。
在非static
成员函数的形参列表后面添加引用限定符(reference qualifier)可以指定this
的左值/右值属性。引用限定符可以是&
或者&&
,分别表示this
可以指向一个左值或右值对象。引用限定符必须同时出现在函数的声明和定义中。
class Foo
{
public:
Foo &operator=(const Foo&) &; // may assign only to modifiable lvalues
// other members of Foo
};
Foo &Foo::operator=(const Foo &rhs) &
{
// do whatever is needed to assign rhs to this object
return *this;
}
一个非static
成员函数可以同时使用const
和引用限定符,此时引用限定符跟在const
限定符之后。
class Foo
{
public:
Foo someMem() & const; // error: const qualifier must come first
Foo anotherMem() const &; // ok: const qualifier comes first
};
引用限定符也可以区分成员函数的重载版本。
class Foo
{
public:
Foo sorted() &&; // may run on modifiable rvalues
Foo sorted() const &; // may run on any kind of Foo
};
retVal().sorted(); // retVal() is an rvalue, calls Foo::sorted() &&
retFoo().sorted(); // retFoo() is an lvalue, calls Foo::sorted() const &
如果一个成员函数有引用限定符,则具有相同参数列表的所有重载版本都必须有引用限定符。