当前位置: 首页 > news >正文

单例模式

文章目录

  • 单例模式
    • 特殊类的设计
    • 单例模式
      • 饿汉模式
      • 懒汉模式
      • 懒汉VS饿汉
      • 懒汉的线程安全
      • 单例对象的释放

单例模式

认识单例模式之前先认识一下几种常见的特殊类的设计。

特殊类的设计

  • 设计一个类 只能再堆上创建对象

    • 只能再堆上创建,则通过new来创建对象。

      1. 将类的构造函数设为私有,以防止外部直接创建对象。
      2. 提供一个静态成员函数,用于在堆上创建对象,并返回指向该对象的指针
      3. 在类的析构函数中释放对象的内存,以确保对象在不再需要时被正确销毁。
     class HeapOnly {public:// 获取堆上对象的静态成员函数static HeapOnly* CreateObj() {return new HeapOnly();}// 删除拷贝构造函数和赋值运算符,以确保只能通过 create() 创建对象HeapOnly(const HeapOnly&) = delete;HeapOnly& operator=(const HeapOnly&) = delete;private:// 将构造函数设为私有,以防止外部直接创建对象HeapOnly() {// 可以在这里进行初始化操作}// 在析构函数中释放对象的内存~HeapOnly() {// 可以在这里进行资源释放操作}};
    • HeapOnly 类的构造函数是私有的,外部无法直接调用。

    • 向外提供的CreateObj方法必须定义为static成员函数。因为外部调用该接口就是为了获取对象的,而非静态成员函数必须通过对象才能调用,静态成员函数直接使用类名::函数名即可调用。

    • 将拷贝构造函数和赋值运算符重载函数之间禁用,也可以设置为私有属性,以防通过拷贝和赋值创建对象。

    • 使用示例:

      int main() {// 创建堆上的对象HeapOnly* obj = HeapOnly::create();// 使用对象...// 删除对象delete obj;return 0;
      }
  • 设计一个类 只能再栈上创建对象

    • 只能再栈上创建对象,说明不能通过new创建对象,也不能定义为static对象。创建方式如下:

      1. 将类的构造函数设为私有,防止外部直接调用构造函数在堆上创建对象。
      2. 在类中定义一个静态成员函数,用于创建对象,并返回该对象的引用或指针。
      class StackOnly {
      public:// 获取栈上对象的静态成员函数static StackOnly& create() {// 在静态成员函数中创建对象,返回对象的引用static StackOnly instance;return instance;}// 删除拷贝构造函数和赋值运算符,以防止通过拷贝或赋值创建对象StackOnly(const StackOnly&) = delete;StackOnly& operator=(const StackOnly&) = delete;private:// 将构造函数设为私有,防止外部直接创建对象StackOnly() {// 可以在这里进行初始化操作}};
      int main() {// 创建栈上的对象StackOnly& obj = StackOnly::create();return 0;
      }
    • StackOnly类的构造函数私有化,外部无法直接调用。限制了通过new创建对象和static对象。

    • 静态局部对象的引用是存储在栈上的。

    • 将拷贝构造函数和赋值运算符重载函数之间禁用,也可以设置为私有属性,以防通过拷贝和赋值创建对象。

  • 设计一个类 不能被拷贝

    • 将拷贝构造函数和赋值运算符设置为私有,或者直接使用delete禁用即可。
    class NonCopyable {
    public:NonCopyable() {} // 默认构造函数// 禁用拷贝构造函数和拷贝赋值运算符NonCopyable(const NonCopyable&) = delete;NonCopyable& operator=(const NonCopyable&) = delete;
    };int main() {NonCopyable obj1;// NonCopyable obj2 = obj1; // 这行会导致编译错误,因为拷贝构造函数被删除了// NonCopyable obj3;// obj3 = obj1; // 这行会导致编译错误,因为拷贝赋值运算符被删除了return 0;
    }
    
  • 设计一个类 不能被继承

    • 将该类的构造函数设置为私有即可,因为子类对象构造时,必须先调用父类构造,构造父类那一部分成员。将其私有化后,子类对象创建对象时,无法调用父类构造函数进行初始化。

      class NonInherit
      {
      public:static NonInherit CreateObj(){return NonInherit();}
      private://将构造函数设置为私有NonInherit(){}
      };
      
    • 也可以使用C++11提供的关键字final。别final修饰的类,无法被继承

      class NonInherit final
      {//...
      };
  • 设计一个类 只能创建一个对象

    • 一个类只能创建一个对象,将其称之为单例模式。

单例模式

单例模式是一种设计模式(Design Pattern),设计模式就是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式的目的就是为了可重用代码、让代码更容易被他人理解、保证代码可靠性程序的重用性。

单例模式指的就是一个类只能创建一个对象,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。

单例模式有两种实现方式,分别是饿汉模式和懒汉模式

饿汉模式

单例模式的实现方式:

  1. 私有化构造函数:确保其他类无法直接实例化该类,只能通过指定的方法获取类的实例。
  2. 私有静态成员变量:保存类的唯一对象。并在程序入口之前完成单例对象的初始化。
  3. 公有静态方法:提供对唯一实例的访问。
class Singleton
{
public:	//公有静态方法static Singleton* GetInstace(){return _inst;}//禁用拷贝构造和赋值运算符Singleton(const Singleton& s) = delete;Singleton& operator=(const Singleton) = delete;
private://私有化构造函数Singleton(){}//私有静态成员变量static Singleton* _inst;
};
Singleton* Singleton::_inst = new Singleton;//静态成员变量的初始化int main()
{//获取单例对象Singleton* ojb = Singleton::GetInstace();
}

懒汉模式

懒汉模式的实现方式:

  1. 私有化构造函数:确保其他类无法直接实例化该类,只能通过指定的方法获取类的实例。
  2. 私有静态成员变量:保存类的唯一对象,将其初始化为空。
  3. 公有静态方法:提供对唯一实例的访问。如果实例不存在,则创建一个新实例并返回;如果实例已存在,则返回现有实例
// 懒汉模式
class Singleton
{
public://公有静态方法static Singleton* GetInstance(){if (_inst == nullptr){_inst = new Singleton;}return _inst;}//禁用拷贝构造和赋值运算符Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
private://私有构造函数Singleton(){}//私有静态成员变量static Singleton* _inst;
};
Singleton* Singleton::_inst = nullptr;int main()
{//获取单例对象Singleton* obj = Singleton::GetInstance();return 0;
}

懒汉VS饿汉

  • 饿汉
    1. 在类加载时就创建实例。无论是否使用该实例,都会在类加载时被创建。如果这个类很大,会影响程序的启动速度。
    2. 在mian程序加载之前就会被实例化,不存在线程安全问题
    3. 使用简单
  • 懒汉
    1. 在需要时才创建实例。也就是说,实例的创建延迟到第一次使用时才发生。
    2. 在多线程环境下需要考虑线程安全性,特别是在第一次获取实例时,可能会存在多个线程同时创建实例的问题。需要通过加锁或者双重检查锁定等技术来确保线程安全性。
    3. 使用复杂

懒汉的线程安全

上面写的懒汉单例并不是线程安全的,在多线程场景下,存在线程安全的问题。GetInstance函数第一次调用时,需要对_inst进行写入操作,这个操作不是线程安全的,多个线程可能同时调用GetInstance进行写入,如果不对此过程进行加锁保护,多线程下就会各自创建出一个对象。

对于饿汉模式来讲,不存在线程安全的问题,因为在main函数之前就已经创建好对象了。

GetInstance中创建单例对象进行加锁保护。

static Singleton* GetInstance()
{if (_inst == nullptr){_mtx.lock();if (_inst == nullptr){_inst = new Singleton;}_mtx.unlock();}return _inst;
}

进行双重检查加锁来保证线程安全。双重检查会提高效率,如果是单检查,如下:

static Singleton* GetInstance()
{_mtx.lock();if (_inst == nullptr){_inst = new Singleton;}_mtx.unlock();return _inst;
}

这样每次访问单例对象时都要进行加锁,加锁之后才能进行判断。双检查之后,只有第一次访问单例对象时才需要加锁。后续访问时无需加锁了。提高了效率。

单例对象的释放

单例对象创建后一般在整个程序运行期间都可能会使用,所以我们可以不考虑单例对象的释放,程序正常结束时会自动将资源归还给操作系统。

有些场景下可能需要提前释放单例对象,可以参考一下方式:

  • 提供一个公有的DelInstance函数,在该函数中进行单例对象释放的操作,当不需要单例对象时,调用此函数进行单例对象的释放。函数如下:
static void DelInstance()
{_mtx.lock();if (_inst != nullptr){delete _inst;_inst = nullptr;}_mtx.unlock();
}

释放操作只能调用一次,单检查加锁足够了,无需双检查加锁。

注意:单例模式中的单例对象和互斥锁,和提供的公有方法都是static的。

首先要明确static修饰成员变量和成员函数的特征:

  1. 静态成员变量

    • 所有类的对象共享静态成员变量的值。
    • 在类声明中声明为静态成员变量,但在类外部必须在类外进行定义和初始化。
    • 静态成员变量的内存只分配一次,直到程序结束才会被释放。
    • 可以在类外部通过类名和作用域解析运算符(::)来访问静态成员变量。
  2. 静态成员函数

    • 不与任何特定的对象相关联,可以直接通过类名调用。
    • 静态成员函数没有隐含的this指针。
    • 静态成员函数不能访问非静态成员变量和非静态成员函数(除非它们是同一个类中的静态成员)。
    • 由于它们不与特定对象相关联,因此无法在静态成员函数中使用this指针。

在单例模式中,需要将实例保存在静态成员变量中,是因为:

  1. 全局可访问性:静态成员变量属于类而不是对象,因此可以被该类的所有对象访问。这符合单例模式的要求,即只有一个实例,并且可以在任何地方访问到该实例。
  2. 生命周期与类相同:静态成员变量随着类的加载而初始化,而不是随着对象的创建而初始化。这意味着无论是否存在对象,静态成员变量都会在类加载时创建,从而保证了在整个应用程序生命周期内只有一个实例存在。

因此,将单例实例保存在静态成员变量中能够满足单例模式的要求,确保了实例的全局可访问性和唯一性。

相关文章:

单例模式

文章目录 单例模式特殊类的设计单例模式饿汉模式懒汉模式懒汉VS饿汉懒汉的线程安全单例对象的释放 单例模式 认识单例模式之前先认识一下几种常见的特殊类的设计。 特殊类的设计 设计一个类 只能再堆上创建对象 只能再堆上创建,则通过new来创建对象。 将类的构造函…...

Android OpenMAX - 开篇

Android Media是一块非常庞大的内容,上到APP的书写,中到播放器的实现、封装格式的了解,下到OMX IL层的实现、Decoder的封装,每一块都需要我们下很大的功夫学习。除此之外,我们还要对一些相关的模块进行了解&#xff0c…...

ubuntu开启ssh服务

1.安装openssh-server sudo apt-get install openssh-server 2.开启服务 sudo servive ssh start 3.设置开机自启动 sudo systemctl enable ssh 4.检查是否开启成功 ps -ef | grep ssh 如使用xshell等连接时失败,可以尝试关闭防火墙: sudo sys…...

测试缺陷定位的基本方法

前后端bug特征 后端: 业务逻辑问题:如任务状态未扭转成功,创建任务失败等数据类问题:如新增的任务在页面没有展示出来等性能类问题:提交任务一直显示创建中、批量操作等待耗时长超时等 前端: 页面显示类…...

【数字图像处理matlab系列】数组索引

【数字图像处理matlab系列】数组索引 【先赞后看养成习惯】【求点赞+关注+收藏】 MATLAB 支持大量功能强大的索引方案,这些索引方案不仅简化了数组操作,而且提高了程序的运行效率。 1. 向量索引 维数为1xN的数组称为行向量。行向量中元素的存取是使用一维索引进行的。因此…...

【2024系统架构设计】案例分析- 3 数据库

目录 一 基础知识 二 真题 一 基础知识 1 ORM ORM(Object—Relationl Mapping...

vue基础——java程序员版(总集)

前言: ​ 这是一个java程序员的vue学习记录。 ​ vue是前端的主流框架,按照如今的就业形式作为后端开发的java程序员也是要有所了解的,下面是本人的vue学习记录,包括vue2的基本使用以及引入element-ui,使用的开发工具…...

Rancher(v2.6.3)——Rancher配置Harbor镜像仓库

Rancher配置Harbor镜像仓库详细说明文档:https://gitee.com/WilliamWangmy/snail-knowledge/blob/master/Rancher/Rancher%E4%BD%BF%E7%94%A8%E6%96%87%E6%A1%A3.md#8rancher%E9%85%8D%E7%BD%AEharbor ps:如果觉得作者写的还行,能够满足您的需…...

C++类和对象、面向对象编程 (OOP)

文章目录 一、封装1.抽象、封装2.类和对象(0)学习视频(1)类的构成(2)三种访问权限(3)struct和class的区别(4)私有的成员变量、共有的成员函数(5)类内可以直接访问私有成员,不需要经过对象 二、继承三、多态1.概念2.多态的满足条件3.多态的使用条件4.多态原理剖析5.纯…...

Introduction to Data Mining 数据挖掘

Why Data Mining? • The Explosive Growth of Data: from terabytes to petabytes — Data collection and data availability ◦ Automated data collection tools, database systems, Web, computerized society — Major sources of abundant data ◦ Business: Web, e-co…...

常用的 Git 命令

初始化一个新的仓库&#xff1a; git init 克隆一个仓库&#xff1a; git clone <仓库地址> 查看文件状态&#xff1a; git status 添加文件到暂存区&#xff1a; git add <文件名> 提交文件到仓库&#xff1a; git commit -m "提交说明" 查看提交历…...

jackson:JSON字符串(String)类型的成员序列化和反序列化

对于如下类型定义TestTaskInfo&#xff0c;props字段为JSON字符串(这在数据库经常用到),可以自由保存各种类型的数据 Data public class TestTaskInfo {private String id;private String props;public TestTaskInfo() {}public TestTaskInfo(String id, String props) {super…...

使用docker的好处???(docker的优势)

标准化环境&#xff1a; Docker通过容器技术封装应用程序及其依赖&#xff08;如库、配置文件、运行时环境等&#xff09;&#xff0c;确保应用程序在任何环境中都能以一致的方式运行。这种标准化消除了“在我机器上能运行”的问题&#xff0c;因为容器化应用能在开发、测试、生…...

Google AI 肺癌筛查的计算机辅助诊断

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…...

【字符串算法题记录】反转字符串中的单词(leetcode),右旋字符串(kama)——双指针以及反转的奇思妙用

反转字符串中的单词 题目链接 思考 这题的思路顺序是&#xff1a;移除多余空格&#xff08;双指针法&#xff09;——》反转整个字符串&#xff09;——》反转字符串中每个单词。 移除多余空格&#xff08;双指针法&#xff09; 因为字符串开头也可能有多个字符&#xff0…...

基于springboot+vue调用百度ai实现车牌号识别功能

百度车牌号识别官方文档 结果视频演示 后端代码 private String getCarNumber(String imagePath, int count) {// 请求urlString url "https://aip.baidubce.com/rest/2.0/ocr/v1/license_plate";try {byte[] imgData FileUtil.readFileByBytes(imagePath);Stri…...

【NTN 卫星通信】 TN和多NTN配合的应用场景

1 场景描述 此场景描述了农村环境&#xff0c;其中MNO (运营商TerrA)仅在城市附近提供本地地面覆盖&#xff0c;而MNO (SatA)提供广泛的NTN覆盖。SatA使用GSO轨道和NGSO轨道上的卫星。SatA与TerrA有漫游协议&#xff0c;允许:   所有TerrA用户的连接&#xff0c;当这些用户不…...

健康餐饮必备!油烟净化器超强洁净餐饮环境

我最近分析了餐饮市场的油烟净化器等产品报告&#xff0c;解决了餐饮业厨房油腻的难题&#xff0c;更加方便了在餐饮业和商业场所有需求的小伙伴们。 ​在如今注重健康生活的时代&#xff0c;餐饮业不仅需要美味佳肴&#xff0c;更需要一个清洁、舒适的用餐环境。油烟净化器作…...

Redis修改开源协议,6大备胎重见天日

背景&#xff1a;Redis2018年以来修改了多次开源协议&#xff0c;以前是把一些高级功能收费&#xff0c;这次彻底怒了&#xff0c;把核心代码的协议修改为RSALv2和SSPL双重协议&#xff0c;这个修改对普通用户不受影响&#xff0c;是向所有云厂商开炮&#xff0c;以后云厂商将不…...

使用python读取csv文件快速插入postgres数据库

使用python读取csv文件快速插入postgres数据库 下面为完整代码 import pandas as pd import cStringIO import warnings from sqlalchemy import create_engine import sys reload(sys) sys.setdefaultencoding(utf8) warnings.filterwarnings(ignore) engine create…...

C++实现分布式网络通信框架RPC(3)--rpc调用端

目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中&#xff0c;我们已经大致实现了rpc服务端的各项功能代…...

在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能

下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能&#xff0c;包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

uniapp微信小程序视频实时流+pc端预览方案

方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度​WebSocket图片帧​定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐​RTMP推流​TRTC/即构SDK推流❌ 付费方案 &#xff08;部分有免费额度&#x…...

select、poll、epoll 与 Reactor 模式

在高并发网络编程领域&#xff0c;高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表&#xff0c;以及基于它们实现的 Reactor 模式&#xff0c;为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。​ 一、I…...

ip子接口配置及删除

配置永久生效的子接口&#xff0c;2个IP 都可以登录你这一台服务器。重启不失效。 永久的 [应用] vi /etc/sysconfig/network-scripts/ifcfg-eth0修改文件内内容 TYPE"Ethernet" BOOTPROTO"none" NAME"eth0" DEVICE"eth0" ONBOOT&q…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲

文章目录 前言第一部分&#xff1a;体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分&#xff1a;体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...

莫兰迪高级灰总结计划简约商务通用PPT模版

莫兰迪高级灰总结计划简约商务通用PPT模版&#xff0c;莫兰迪调色板清新简约工作汇报PPT模版&#xff0c;莫兰迪时尚风极简设计PPT模版&#xff0c;大学生毕业论文答辩PPT模版&#xff0c;莫兰迪配色总结计划简约商务通用PPT模版&#xff0c;莫兰迪商务汇报PPT模版&#xff0c;…...

Chromium 136 编译指南 Windows篇:depot_tools 配置与源码获取(二)

引言 工欲善其事&#xff0c;必先利其器。在完成了 Visual Studio 2022 和 Windows SDK 的安装后&#xff0c;我们即将接触到 Chromium 开发生态中最核心的工具——depot_tools。这个由 Google 精心打造的工具集&#xff0c;就像是连接开发者与 Chromium 庞大代码库的智能桥梁…...

0x-3-Oracle 23 ai-sqlcl 25.1 集成安装-配置和优化

是不是受够了安装了oracle database之后sqlplus的简陋&#xff0c;无法删除无法上下翻页的苦恼。 可以安装readline和rlwrap插件的话&#xff0c;配置.bahs_profile后也能解决上下翻页这些&#xff0c;但是很多生产环境无法安装rpm包。 oracle提供了sqlcl免费许可&#xff0c…...

C++实现分布式网络通信框架RPC(2)——rpc发布端

有了上篇文章的项目的基本知识的了解&#xff0c;现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...