【C++入门到精通】特殊类的设计 |只能在堆 ( 栈 ) 上创建对象的类 |禁止拷贝和继承的类 [ C++入门 ]

简介: 【C++入门到精通】特殊类的设计 |只能在堆 ( 栈 ) 上创建对象的类 |禁止拷贝和继承的类 [ C++入门 ]

引言

在面向对象编程中,特殊类是指具有不同于常规类的特殊属性或限制的类。这些类可以通过各种方式达到特定的目标和需求,例如只能在堆 ( 栈 ) 上创建对象的类、禁止拷贝和继承等。

本文将会讨论四种常见的特殊类:不能被拷贝的类、只能在堆上创建对象的类、只能在栈上创建对象的类以及不能被继承的类。我们将介绍它们的实现方法和应用场景,并提供相应的代码示例以帮助读者更好地理解这些特殊类的概念和用法。

一、特殊类 — 不能被拷贝的类

当一个类的拷贝是不允许的,可以采取以下两种方式来实现:

1. C++98方式:

在C++98中,可以通过将拷贝构造函数和赋值运算符重载声明为私有,并且不实现它们来禁止拷贝。这样做的原因是私有访问权限限制了类外部的代码无法访问这两个函数的实现

class CopyBan {
private:
    CopyBan(const CopyBan&); // 声明拷贝构造函数为私有
    CopyBan& operator=(const CopyBan&); // 声明赋值运算符重载为私有
public:
    // 其他公共成员函数和数据成员
};

这种方式通过将拷贝构造函数和赋值运算符重载放在私有区域,从而阻止了类的外部对象调用这两个函数,实现了禁止拷贝的目的。

2. C++11方式:

在C++11中,可以使用delete关键字来删除拷贝构造函数和赋值运算符重载。通过在函数声明后加上= delete,可以明确告诉编译器要删除此函数,防止类外部的代码调用它

class CopyBan {
public:
    CopyBan(const CopyBan&) = delete; // 删除拷贝构造函数
    CopyBan& operator=(const CopyBan&) = delete; // 删除赋值运算符重载
    // 其他公共成员函数和数据成员
};

这种方式更加简洁明了,直观地表达了禁止拷贝的意图。使用= delete语法可以方便地阻止拷贝构造函数和赋值运算符重载的调用。

无论是C++98还是C++11,这两种方式都能有效地禁止类的拷贝,增强代码的稳定性和安全性。选择哪种方式取决于你使用的C++版本和个人偏好。

二、特殊类 — 只能在堆上创建对象的类

要设计一个只能在堆上创建对象的类,可以使用私有的析构函数和静态成员函数来实现:

class HeapOnly {
private:
    HeapOnly() {} // 私有的默认构造函数,防止在栈上创建对象
    ~HeapOnly() {} // 私有的析构函数,防止在栈上销毁对象

public:
    static HeapOnly* createInstance() {
        return new HeapOnly();
    }

    void destroyInstance() {
        delete this;
    }
};

这个类中,私有的默认构造函数和析构函数阻止了在栈上创建和销毁对象。而通过静态的createInstance()函数,可以在堆上创建该类的对象,并返回指向该对象的指针。然后,我们可以使用对象的destroyInstance()函数在适当的时候手动删除对象,从而释放内存。

以下是示例代码的使用方式:

int main() {
    HeapOnly* obj = HeapOnly::createInstance();

    // 使用对象

    obj->destroyInstance();

    return 0;
}

🚨🚨注意:在这种设计中,由于析构函数是私有的,不能直接使用delete操作符来销毁对象。只能通过调用对象的destroyInstance()函数来手动删除对象

三、特殊类 — 只能在栈上创建对象的类

要设计一个只能在栈上创建对象的类,可以使用私有的构造函数和公有的静态成员函数来实现:

class StackOnly {
public:
    static StackOnly createInstance() {
        return StackOnly{};
    }

private:
    StackOnly() = default; // 私有的默认构造函数,防止在堆上创建对象
    ~StackOnly() = default; // 默认析构函数
};

这个类中,私有的构造函数阻止了在堆上创建对象。而通过公有的静态createInstance()函数,可以在栈上创建该类的对象,并返回该对象的副本由于没有指针和动态内存分配,对象的生命周期由对象所在的作用域控制,当对象离开作用域时,会自动调用析构函数销毁对象

以下是示例代码的使用方式:

int main() {
    StackOnly obj = StackOnly::createInstance();

    // 使用对象

    return 0;
}

这样设计的类限制了对象只能在栈上创建,可以避免使用者误用动态内存分配,从而提高了代码的健壮性。但请注意,在这种设计中,对象的拷贝构造函数、赋值运算符重载函数需要适当处理,以确保对象的正确复制行为。

四、特殊类 — 不能被继承的类

1. C++98方式

// C++98中构造函数私有化,派生类中调不到基类的构造函数。则无法继承
class NonInherit
{
public:
  static NonInherit GetInstance()
  {
    return NonInherit();
  }
private:
  NonInherit()
  {}
};

🚨🚨注意:在C++98中,虽然可以将构造函数私有化以阻止派生类调用基类的构造函数,但这并不能完全阻止继承。派生类仍然可以继承基类的成员函数和非私有成员变量

2. C++11方法

要设计一个不能被继承的类,可以使用C++中的关键字final来实现:

class NonInheritable final {
    // 类的定义
};

通过在类的声明中添加final关键字,可以阻止其他类继承该类。这样设计的类将是最终类,不能被其他类所继承。

以下是示例代码的使用方式:

class Derived : public NonInheritable { // 编译错误,无法继承NonInheritable类

};

int main() {
    NonInheritable obj; // 创建NonInheritable类的对象

    // 使用对象

    return 0;
}

这样设计的类不能被继承,可以确保类的封装性和稳定性,防止其他类对其进行修改或破坏。这在某些情况下非常有用,特别是当你希望限制类的继承性时。

总结

这些特殊类在面向对象编程中有着重要的作用,并且通过选择合适的类设计模式,我们可以更好地满足需求和解决问题。

在下一篇文章中,我们将深入研究单例模式。单例模式是一种常用的设计模式,它确保一个类只有一个实例,并提供了全局访问点。当我们需要确保只有一个对象来协调系统操作或管理共享资源时,单例模式非常有用。通过深入学习单例模式,我们将能够更好地理解其原理和使用方法,以及如何在实际开发中应用它。单例模式是一种非常有用的设计模式,掌握它将有助于我们编写可靠、高效的代码。让我们一起探索单例模式的精髓吧!敬请期待下一篇文章的发布!

温馨提示

感谢您对博主文章的关注与支持!另外,我计划在未来的更新中持续探讨与本文相关的内容,会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!

再次感谢您的支持和关注。期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

目录
相关文章
|
28天前
|
编译器 C++ 开发者
【C++】继承
C++中的继承是面向对象编程的核心特性之一,允许派生类继承基类的属性和方法,实现代码复用和类的层次结构。继承有三种类型:公有、私有和受保护继承,每种类型决定了派生类如何访问基类成员。此外,继承还涉及构造函数、析构函数、拷贝构造函数和赋值运算符的调用规则,以及解决多继承带来的二义性和数据冗余问题的虚拟继承。在设计类时,应谨慎选择继承和组合,以降低耦合度并提高代码的可维护性。
32 1
【C++】继承
|
25天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
40 2
|
1月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
83 5
|
1月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
80 4
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
88 4
|
2月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
31 4
|
2月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
32 4
|
2月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
26 1
|
2月前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
2月前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)