Explicitly disallow the use of compiler-generated functions you do not want.
通常如果你不希望class支持某一特定的机能,只要不声明对应函数就是了。但这个策略对copy构造函数和copy assignment操作符却不起作用,因为如果你不声明它们,而某些人尝试调用它们,编译器会为你声明它们。
答案的关键是,所有编译器产出的函数都是public。为阻止这些函数被创建出来,你得自行声明它们,但这里并没有什么需求使你必须将它们声明为public。因此你可以将copy构造函数或者copy assignment操作符声明为private。藉由明确声明一个成员函数,你阻止了编译器暗自创建其专属版本;而令这些函数为private,使你得以成功阻止人们调用它。
一般而言这个做法并不绝对安全,因为成员函数和友元函数函数可以调用你的private函数。除非你够聪明,不去定义它们,那么如果某些人不慎调用任何一个,会获得一个连接错误。将连接期错误移至编译期是可能,需要设计一个专门为了阻止copying动作而设计的base class内,这个base class非常简单:
class Uncopyable{
protected:
Uncopyable(){}
~Uncopyable(){}
private:
Uncopyable(const Uncopyable &);
Uncopyable &operator=(const Uncopyable &);
}
class HomeForSale:private Uncopyable {
}
只要像HomeForSale类这样继承Uncopyable就可以阻止对象被拷贝。只要任何人尝试拷贝HomeForSale对象,编译器便试着生成一个copy构造函数和一个copy assignment操作符,这些函数的“编译器生成版”会尝试调用其base class的对应兄弟,那些调用会被编译器拒绝,因为其base class的拷贝函数是private。
请记住:
- 为驳回编译器自动提供的机能,可将相应的成员函数声明为private并且不予实现。使用像Uncopyable这样的base class也是一种做法。