C++ 内存管理与单例模式剖析

news2025/5/25 14:59:10

目录

引言

一、堆上唯一对象:HeapOnly类

(一)设计思路

(二)代码实现

(三)使用示例及注意事项

二、栈上唯一对象:StackOnly类

(一)设计思路

(二)代码实现

(三)使用示例及注意事项

三、单例模式:饿汉模式与懒汉模式

(一)单例模式概述

(二)饿汉模式

(三)懒汉模式

(四)单例模式使用示例

总结


引言

在C++ 编程中,内存管理和设计模式是非常重要的两个方面。合理的内存管理能确保程序高效、稳定地运行,而设计模式则有助于构建更具可维护性、可扩展性的软件架构。今天,我们将深入探讨C++ 中关于内存管理的一些特殊类设计,以及经典的单例模式。

一、堆上唯一对象:HeapOnly类

(一)设计思路

 HeapOnly  类的设计目的是强制对象只能在堆上创建。这是通过将构造函数设为私有来实现的。外部代码无法直接调用构造函数在栈上创建对象,也不能使用  static  关键字在静态存储区创建对象。

(二)代码实现

class HeapOnly
{
public:
    static HeapOnly* CreateObj()
    {
        return new HeapOnly;
    }
private:
    HeapOnly()
    {
        //...
    }
    HeapOnly(const HeapOnly& hp) = delete;
    HeapOnly& operator=(const HeapOnly& hp) = delete;
};

这里,唯一能创建  HeapOnly  对象的方式是通过静态成员函数  CreateObj  ,它使用  new  操作符在堆上分配内存并构造对象。同时,将拷贝构造函数和赋值运算符重载函数设为删除状态,防止对象被拷贝,进一步保证对象的唯一性和内存管理的安全性。

(三)使用示例及注意事项

//int main()

//{
// //HeapOnly hp1; // 错误,无法在栈上创建
// //static HeapOnly hp2; // 错误,无法在静态存储区创建
// //HeapOnly* hp3 = new HeapOnly; // 错误,构造函数私有
// HeapOnly* hp3 = HeapOnly::CreateObj();
// HeapOnly copy(*hp3); // 错误,拷贝构造函数被删除
// return 0;
//}

在使用时,要严格遵循其设计规则,只能通过  CreateObj  获取对象指针,并且不能进行拷贝操作。

二、栈上唯一对象:StackOnly类

(一)设计思路

 StackOnly  类与  HeapOnly  类相反,它的设计是为了确保对象只能在栈上创建。通过将  operator new  设为删除状态,禁止了使用  new  操作符在堆上创建对象。

(二)代码实现

class StackOnly
{
public:
    static StackOnly CreateObj()
    {
        StackOnly st;
        return st;
    }
private:
    StackOnly()
    {
        //...
    }
    void* operator new(size_t size) = delete;

};

 CreateObj  函数在函数内部的栈空间上创建  StackOnly  对象,并返回该对象的副本。由于  operator new  被删除,无法在堆上创建对象。

(三)使用示例及注意事项

int main()
{
   //StackOnly hp1; // 错误,构造函数私有
    //static StackOnly hp2; // 错误,构造函数私有
    //StackOnly* hp3 = new StackOnly; // 错误,operator new被删除
    StackOnly hp3 = StackOnly::CreateObj();
    StackOnly copy(hp3); // 这里如果类没有合适的拷贝构造函数会有问题
    // new operator new + 构造
    // StackOnly* hp4 = new StackOnly(hp3); // 错误,operator new被删除
    return 0;

}

使用时要注意只能通过  CreateObj  来获取对象,并且要确保类的拷贝构造函数等符合需求,避免出现意外的错误。

三、单例模式:饿汉模式与懒汉模式

(一)单例模式概述

单例模式是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点来访问这个实例。在很多场景下,比如日志记录器、数据库连接池等,只需要一个全局唯一的对象来进行管理和操作,单例模式就能很好地满足这种需求。

(二)饿汉模式

1. 设计思路:饿汉模式在程序启动( main  函数之前)就创建单例对象。无论后续是否会用到这个单例对象,它都会被提前创建。

2. 代码实现

namespace hungry
{
    class Singleton
    {
    public:
        static Singleton& GetInstance()
        {
            return _sinst;
        }
        void func();
        void Add(const pair<string, string>& kv)
        {
            _dict[kv.first] = kv.second;
        }
        void Print()
        {
            for (auto& e : _dict)
            {
                cout << e.first << ":" << e.second << endl;
            }
            cout << endl;
        }
    private:
        Singleton()
        {
            // ...
        }
        Singleton(const Singleton& s) = delete;
        Singleton& operator=(const Singleton& s) = delete;
        map<string, string> _dict;
        static Singleton _sinst;
    };
    Singleton Singleton::_sinst;
    void Singleton::func()
    {
        _dict["xxx"] = "1111";
    }

}

这里, _sinst  是静态成员变量,在类外进行了定义和初始化。 GetInstance  函数返回这个唯一的单例对象的引用。

3. 优缺点

优点:实现简单,在程序启动时就创建好对象,不存在多线程并发创建对象的问题。

缺点:如果单例对象初始化内容很多,会影响程序的启动速度。并且当有多个互相依赖的单例类时,难以保证初始化顺序。

(三)懒汉模式

1. 设计思路:懒汉模式是在第一次调用获取单例对象的函数时才创建对象。这样可以避免在程序启动时就创建不必要的对象,提高程序的启动效率。

2. 代码实现

namespace lazy
{
    class Singleton
    {
    public:
        static Singleton& GetInstance()
        {
            if (_psinst == nullptr)
            {
                _psinst = new Singleton;
            }
            return *_psinst;
        }

        static void DelInstance()
        {
            if (_psinst)
            {
                delete _psinst;
                _psinst = nullptr;
            }
        }

        void Add(const pair<string, string>& kv)
        {
            _dict[kv.first] = kv.second;
        }

        void Print()
        {
            for (auto& e : _dict)
            {
                cout << e.first << ":" << e.second << endl;
            }
            cout << endl;
        }

        class GC
        {
        public:
            ~GC()
            {
                lazy::Singleton::DelInstance();
            }
        }
    private:
        Singleton()
        {
            // ...
        }
        ~Singleton()
        {
            cout << "~Singleton()" << endl;
            FILE* fin = fopen("map.txt", "w");
            for (auto& e : _dict)
            {
                fputs(e.first.c_str(), fin);
                fputs(":", fin);
                fputs(e.second.c_str(), fin);
                fputs("\n", fin);
            }
        }
        Singleton(const Singleton& s) = delete;
        Singleton& operator=(const Singleton& s) = delete;
        map<string, string> _dict;
        static Singleton* _psinst;
        static GC _gc;
    };
    Singleton* Singleton::_psinst = nullptr;
    Singleton::GC Singleton::_gc;
}

这里通过  GetInstance  函数中的  if  判断来实现延迟创建对象。同时,定义了一个内部类  GC  ,利用其析构函数在程序结束时释放单例对象,确保资源的正确回收。

3. 优缺点

优点:延迟创建对象,提高启动速度,并且可以在程序运行中根据需要释放单例对象,在一些特殊场景(如中途需要释放资源或程序结束时做持久化操作)下很有用。

缺点:在多线程环境下,如果不进行同步处理,可能会出现多个线程同时创建对象的问题,导致违反单例模式的原则。

(四)单例模式使用示例

int main()
{
    cout << &lazy::Singleton::GetInstance() << endl;
    cout << &lazy::Singleton::GetInstance() << endl;
    cout << &lazy::Singleton::GetInstance() << endl;
    //Singleton copy(Singleton::GetInstance()); // 错误,拷贝构造函数被删除
    lazy::Singleton::GetInstance().Add({ "xxx", "111" });
    lazy::Singleton::GetInstance().Add({ "yyy", "222" });
    lazy::Singleton::GetInstance().Add({ "zzz", "333" });
    lazy::Singleton::GetInstance().Add({ "abc", "333" });
    lazy::Singleton::GetInstance().Print();
    //lazy::Singleton::DelInstance();
    lazy::Singleton::GetInstance().Add({ "abc", "444" });
    lazy::Singleton::GetInstance().Print();
    //lazy::Singleton::DelInstance();
    return 0;
}

在  main  函数中,多次调用  GetInstance  获取单例对象,并对其进行操作,验证了单例对象的唯一性和可操作性。

总结

通过对  HeapOnly  类、 StackOnly  类以及单例模式的饿汉模式和懒汉模式的深入剖析,我们了解了C++ 中一些特殊的内存管理方式和经典的设计模式。这些知识在实际编程中非常实用,合理运用它们可以让我们的程序在内存管理上更加合理,架构上更加清晰和稳定。在具体应用时,要根据实际需求和场景选择合适的方案,同时注意避免出现内存泄漏、对象创建错误等问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2385414.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

算法学习——从零实现循环神经网络

从零实现循环神经网络 一、任务背景二、数据读取与准备1. 词元化2. 构建词表 三、参数初始化与训练1. 参数初始化2. 模型训练 四、预测总结 一、任务背景 对于序列文本来说&#xff0c;如何通过输入的几个词来得到后面的词一直是大家关注的任务之一&#xff0c;即&#xff1a;…

win10使用nginx做简单负载均衡测试

一、首先安装Nginx&#xff1a; 官网链接&#xff1a;https://nginx.org/en/download.html 下载完成后&#xff0c;在本地文件中解压。 解压完成之后&#xff0c;打开conf --> nginx.config 文件 1、在 http 里面加入以下代码 upstream GY{#Nginx是如何实现负载均衡的&a…

2025电工杯数学建模B题思路数模AI提示词工程

我发布的智能体链接&#xff1a;数模AI扣子是新一代 AI 大模型智能体开发平台。整合了插件、长短期记忆、工作流、卡片等丰富能力&#xff0c;扣子能帮你低门槛、快速搭建个性化或具备商业价值的智能体&#xff0c;并发布到豆包、飞书等各个平台。https://www.coze.cn/search/n…

【日志软件】hoo wintail 的替代

hoo wintail 的替代 主要问题是日志大了以后会卡有时候日志覆盖后&#xff0c;改变了&#xff0c;更新了&#xff0c;hoo wintail可能无法识别需要重新打开。 有很多类似的日志监控软件可以替代。以下是一些推荐的选项&#xff1a; 免费软件 BareTail 轻量级的实时日志查看…

Ollama-OCR:基于Ollama多模态大模型的端到端文档解析和处理

基本介绍 Ollama-OCR是一个Python的OCR解析库&#xff0c;结合了Ollama的模型能力&#xff0c;可以直接处理 PDF 文件无需额外转换&#xff0c;轻松从扫描版或原生 PDF 文档中提取文本和数据。根据使用的视觉模型和自定义提示词&#xff0c;Ollama-OCR 可支持多种语言&#xf…

OpenCV CUDA 模块中图像过滤------创建一个拉普拉斯(Laplacian)滤波器函数createLaplacianFilter()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::cuda::createLaplacianFilter 是 OpenCV CUDA 模块中的一个函数&#xff0c;用于创建一个 拉普拉斯&#xff08;Laplacian&#xff09;滤波器…

图论学习笔记 3

自认为写了很多&#xff0c;后面会出 仙人掌、最小树形图 学习笔记。 多图警告。 众所周知王老师有一句话&#xff1a; ⼀篇⽂章不宜过⻓&#xff0c;不然之后再修改使⽤的时候&#xff0c;在其中找想找的东⻄就有点麻烦了。当然⽂章也不宜过多&#xff0c;不然想要的⽂章也不…

【将WPS设置为默认打开方式】--突然无法用WPS打开文件

1. 点击【开始】——【WPS Office】——【配置工具】&#xff1b; 2. 在出现的弹窗中&#xff0c;点击【高级】&#xff1b; 3. 在“兼容设置”中&#xff0c;将复选框勾上&#xff0c;点击【确定】。

电子人的分水岭-FPGA模电和数电

为什么模电这么难学&#xff1f;一文带你透彻理解模电 ——FPGA是“前期数电&#xff0c;后期模电”的典型代表 在电子工程的世界里&#xff0c;有两门基础课程让无数学生“闻之色变”&#xff1a;数字电路&#xff08;数电&#xff09; 和 模拟电路&#xff08;模电&#xff0…

(6)python爬虫--selenium

文章目录 前言一、初识selenium二、安装selenium2.1 查看chrome版本并禁止chrome自动更新2.1.1 查看chrome版本2.1.2 禁止chrome更新自动更新 2.2 安装对应版本的驱动程序2.3安装selenium包 三、selenium关于浏览器的使用3.1 创建浏览器、设置、打开3.2 打开/关闭网页及浏览器3…

Python之两个爬虫案例实战(澎湃新闻+网易每日简报):附源码+解释

目录 一、案例一&#xff1a;澎湃新闻时政爬取 &#xff08;1&#xff09;数据采集网站 &#xff08;2&#xff09;数据介绍 &#xff08;3&#xff09;数据采集方法 &#xff08;4&#xff09;数据采集过程 二、案例二&#xff1a;网易每日新闻简报爬取 &#xff08;1&#x…

✨ PLSQL卡顿优化

✨ PLSQL卡顿优化 1.&#x1f4c2; 打开首选项2.&#x1f527; Oracle连接配置3.⛔ 关闭更新和新闻 1.&#x1f4c2; 打开首选项 2.&#x1f527; Oracle连接配置 3.⛔ 关闭更新和新闻

python+vlisp实现对多段线范围内土方体积的计算

#在工程中&#xff0c;经常用到计算土方回填、土方开挖的体积。就是在一个范围内&#xff0c;计算土被挖走&#xff0c;或者填多少&#xff0c;这个需要测量挖填前后这个范围内的高程点。为此&#xff0c;我开发一个app&#xff0c;可以直接在autocad上提取高程点&#xff0c;然…

APM32小系统键盘PCB原理图设计详解

APM32小系统键盘PCB原理图设计详解 一、APM32小系统简介 APM32微控制器是国内半导体厂商推出的一款高性能ARM Cortex-M3内核微控制器&#xff0c;与STM32高度兼容&#xff0c;非常适合DIY爱好者用于自制键盘、开发板等电子项目。本文将详细讲解如何基于APM32 CBT6芯片设计一款…

对象存储(Minio)使用

目录 1.安装 MinIO&#xff08;Windows&#xff09; 2.启动minio服务&#xff1a; 3.界面访问 4.进入界面 5.前后端代码配置 1)minio前端配置 2&#xff09;minio后端配置 1.安装 MinIO&#xff08;Windows&#xff09; 官方下载地址&#xff1a;[Download High-Perform…

yolov11使用记录(训练自己的数据集)

官方&#xff1a;Ultralytics YOLO11 -Ultralytics YOLO 文档 1、安装 Anaconda Anaconda安装与使用_anaconda安装好了怎么用python-CSDN博客 2、 创建虚拟环境 安装好 Anaconda 后&#xff0c;打开 Anaconda 控制台 创建环境 conda create -n yolov11 python3.10 创建完后&…

知识宇宙:技术文档该如何写?

名人说&#xff1a;博观而约取&#xff0c;厚积而薄发。——苏轼《稼说送张琥》 创作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 一、技术文档的价值与挑战1. 为什么技术文档如此重要2. 技术文档面临的挑战 二、撰…

技嘉主板怎么开启vt虚拟化功能_技嘉主板开启vt虚拟化教程(附intel和amd开启方法)

最近使用技嘉主板的小伙伴们问我&#xff0c;技嘉主板怎么开启vt虚拟。大多数可以在Bios中开启vt虚拟化技术&#xff0c;当CPU支持VT-x虚拟化技术&#xff0c;有些电脑会自动开启VT-x虚拟化技术功能。而大部分的电脑则需要在Bios Setup界面中&#xff0c;手动进行设置&#xff…

Java 并发编程高级技巧:CyclicBarrier、CountDownLatch 和 Semaphore 的高级应用

Java 并发编程高级技巧&#xff1a;CyclicBarrier、CountDownLatch 和 Semaphore 的高级应用 一、引言 在 Java 并发编程中&#xff0c;CyclicBarrier、CountDownLatch 和 Semaphore 是三个常用且强大的并发工具类。它们在多线程场景下能够帮助我们实现复杂的线程协调与资源控…

PT5F2307触摸A/D型8-Bit MCU

1. 产品概述 ● PT5F2307是一款51内核的触控A/D型8位MCU&#xff0c;内置16K*8bit FLASH、内部256*8bit SRAM、外部512*8bit SRAM、触控检测、12位高精度ADC、RTC、PWM等功能&#xff0c;抗干扰能力强&#xff0c;适用于滑条遥控器、智能门锁、消费类电子产品等电子应用领域。 …