我们都知道,如果不想让一个class具有某一功能,只要不声明具有该功能的函数成员即可。可是对于copy construct 和 copying assignment确不是这样的,因为即使你不声明这两个函数,编译器也会自动为你生成。所以如果你想让class不具有这个功能,必须要做点工作了。
方法是:
1.将copy construct和copying assignment声明为private;
2.不予实现。
这样做的目的是:
1.自己声明可以阻止编译器自动生成相应版本;
2.令其属性为private,就可以阻止人们调用它。
3.不予实现可以在member function和friend function调用copy函数时时,产生一个连接错误。
这样class就不会具有copy功能。c++ iostream 程序库中就是如此copying行为的,比如ios_base, basic_ios和sentry。
class A{
public:
A(){}
~A() {}
private:
A(const A&);
A& operator=(const A&);
};
int main()
{
A a;
A b(a);
return 0;
}
会出现编译错误:阻止了拷贝的发生
error C2248: 'A::A' : cannot access private member declared in class 'A'
连接错误示例如下所示:
class A{
friend A copyA(const A &temp) {return A(temp);}
public:
A(){}
~A(){}
A mycopy(const A& temp){return A(temp);}
private:
A(const A&);
A& operator=(const A&);
};
int main()
{
A a;
A b;
copyA(a);
b.mycopy(a);
return 0;
}
运行后会报连接错误。如何将连接期错误移至编译期呢?
解决方案:声明一个专门为了阻止copying动作设计的base class
class Uncopyable{
protected:
Uncopyable() {}
~Uncopyable() {}
private:
Uncopyable(const Uncopyable &);
Uncopyable& operator=(const Uncopyable &);
};
class A:private Uncopyable{
...
};
这样,任何人(包括member and friend function)尝试拷贝A对象,编译器便会试着生成一个copy construct和copy assignment,这些编译器生成版会尝试调用base class的相应函数,但这些调用会被拒绝,因为base class的copy函数是private。