目录
序言
(一)设计一个不能被拷贝的类
(二)设计一个只能在堆上创建对象的类
(三)设计一个只能在栈上创建对象的类
(四)设计一个不能被继承的类
总结
特殊类设计是指在面向对象编程中,根据特定需求或情况,创建具备特殊功能或属性的类。特殊类设计旨在解决特定问题或满足特殊需求,使代码更加灵活和可扩展。
拷贝只会放生在两个场景中:拷贝构造函数以及赋值运算符重载,因此想要让一个类禁止拷贝,只需让该类不能调用拷贝构造函数以及赋值运算符重载即可。
接下来,我们分别从C++98和C++11的两种场景去看二者是如何实现:
【C++98】:将拷贝构造函数与赋值运算符重载只声明不定义,并且将其访问权限设置为私有即可
class CopyBan
{
private:
CopyBan(const CopyBan&); // 声明拷贝构造函数为私有
CopyBan& operator=(const CopyBan&); // 声明拷贝赋值运算符为私有
public:
CopyBan() {} // 默认构造函数
};
【解释说明】
使用该类时,只需简单地继承它即可:
class MyClass : public CopyBan{
// 类的定义
};
【C++11】:C++11扩展delete的用法,delete除了释放new申请的资源外,如果在默认成员函数后跟上=delete,表示让编译器删除掉该默认成员函数。
class CopyBan
{
//.....
CopyBan(const CopyBan&) = delete; // 删除拷贝构造函数
CopyBan& operator=(const CopyBan&) = delete; // 删除拷贝赋值运算符
//.....
};
【解释说明】
同样的使用该类时,只需简单地继承它即可:
class MyClass : public CopyBan{
// 类的定义
};
【小结】
实现方式:
- 将类的构造函数私有,拷贝构造声明成私有。防止别人调用拷贝在栈上生成对象。
- 提供一个静态的成员函数,在该静态成员函数中完成堆对象的创建。
【C++98】 :
class HeapOnly
{
public:
static HeapOnly* CreateObject()
{
return new HeapOnly();
}
private:
HeapOnly() {} // 私有化默认构造函数
// C++98
// 1.只声明,不实现。因为实现可能会很麻烦,而你本身不需要
// 2.声明成私有
HeapOnly(const HeapOnly&);
};
【C++11】:
class HeapOnly
{
public:
static HeapOnly* CreateObject()
{
return new HeapOnly();
}
private:
HeapOnly() {} // 私有化默认构造函数
// C++11
HeapOnly(const HeapOnly&) = delete;
};
【解释说明】
= delete
来删除复制构造函数,而不仅仅是将其声明为私有;这样,你就可以使用下面的代码在堆上创建 HeapOnly 类对象:
HeapOnly* obj = HeapOnly::CreateObject();
请确保在使用完对象后手动调用 delete
来释放内存:
delete obj;
【小结】
实现方法:
- 要设计一个只能在栈上创建对象的类,可以使用私有的析构函数和公有的静态成员函数来实现。
【C++98】 :
class StackOnly
{
public:
static StackOnly CreateObj() {
return StackOnly();
}
private:
StackOnly() {} // 私有化默认构造函数
~StackOnly() {} // 私有化析构函数
};
【解释说明】
使用代码示例:
StackOnly obj = StackOnly::CreateObj();
delete
释放内存),当对象超出作用域时会自动调用析构函数进行资源的释放。【C++11】:可以使用删除特殊成员函数以及阻止使用new和delete操作符的方式来实现只能在栈上创建对象的类。
class StackOnly
{
public:
StackOnly() = default; // 允许默认构造函数
// 删除拷贝构造函数和赋值运算符函数
StackOnly(const StackOnly&) = delete;
StackOnly& operator=(const StackOnly&) = delete;
// 禁止使用new和delete操作符
void* operator new(size_t) = delete;
void operator delete(void*) = delete;
};
使用代码示例:
StackOnly obj1; // 在栈上创建对象
// 下面的代码将导致编译错误,因为拷贝构造函数被删除
// StackOnly obj2 = obj1;
// 下面的代码将导致编译错误,因为赋值运算符函数被删除
// StackOnly obj3;
// obj3 = obj1;
// 下面的代码将导致编译错误,因为使用了删除的new运算符
// StackOnly* ptr = new StackOnly;
// 下面的代码将导致编译错误,因为使用了删除的delete运算符
// delete ptr;
【解释说明】
new
和delete
来分配和释放对象的内存,从而确保了对象的唯一性和创建方式的限制。除了上述方法之外,还有一种比较奇特的方式,可以用于实现只能在栈上创建对象的类。这种方式是通过定义一个私有化的
operator new
和operator delete
函数来实现,无需删除构造函数和析构函数。
class StackOnly
{
public:
// 在 public 区域声明 operator new 和 operator delete 函数
static void* operator new(size_t size) = delete; // 删除 operator new 函数
static void operator delete(void* ptr) noexcept = delete; // 删除 operator delete 函数
private:
// 私有化所有构造、析构函数,包括默认构造函数
StackOnly() {}
StackOnly(const StackOnly&) {}
~StackOnly() {}
};
【解释说明】
使用该类时,只能通过栈上的对象进行操作:
StackOnly obj;
【注意】
StackOnly
类的对象只能在栈上创建。但是这种方法需要注意的是,在某些情况下,可能会因为需要使用 operator new 和 operator delete 函数而无法编译通过,因此使用时需要慎重考虑。实现方法:
- 在使用 C++98 标准时,可以通过将构造函数和析构函数设为私有,并且不提供公共的静态工厂方法来实现一个不能被继承的类。
【C++98】 :
class NonInherit
{
private:
NonInherit() {}
~NonInherit() {}
public:
// 禁止通过静态工厂方法创建对象
static NonInherit* GetInstance() {
return NULL;
}
};
实现方法:
- 在C++11以及后续的标准中,你可以在类声明的末尾添加关键字
final
,来显式地指示该类是不可被继承的。
【C++11】:
class A final
{
// ....
};
【解释说明】
例如:
class A final
{
//...
};
class B : public A { // 编译错误
// 类定义
};
报错如下:
这些特殊类的设计目的是根据特定的需求和编程场景来确定的;
它们有助于代码的组织、可维护性、可扩展性和重用性。通过合理地设计和使用这些特殊类,可以提高代码的质量、可读性和可靠性。
到此,关于本期特殊类设计便讲解结束了。感谢大家的观看和支持!!!