设计模式-01 设计模式单例模式

设计模式-01 设计模式单例模式

目录

设计模式-01 设计模式单例模式

1定义

2.内涵

3.使用示例

4.具体代码使用实践

5.注意事项

6.最佳实践

7.总结


1 定义


单例模式是一种设计模式,它确保一个类只能被实例化一次。它通过在类内部创建类的唯一实例并提供一个全局访问点来实现这一点。


优点:

全局访问:单例对象可以从应用程序的任何地方访问。
资源节约:它只创建一个对象,从而节省了内存和资源。
线程安全性:如果单例对象正确实现,它可以是线程安全的,这意味着它可以在多线程环境中安全地使用。


2.内涵

单例模式在 C++ 设计模式中具有以下内涵:

封装:单例类的内部状态和实现细节对客户端代码是隐藏的。这使你可以自由地更改类的实现,而无需影响使用它的代码。
全局访问:单例对象可以在应用程序的任何地方访问,这使得它非常适合存储全局状态或提供公共服务。
资源管理:由于单例模式确保只创建一个对象,因此它可以帮助管理资源,例如数据库连接或文件句柄。

3.使用示例

单例模式,有很多种实现方式,下面是几种实现方式的介绍。

延迟加载

class Singleton {
private:static Singleton* instance;Singleton() {}public:static Singleton* getInstance() {if (instance == nullptr) {instance = new Singleton();}return instance;}
};


饿汉式

class Singleton {
private:static Singleton* instance = new Singleton();Singleton() {}public:static Singleton* getInstance() {return instance;}
};

C++11 实现单例模式代码

class Singleton {
private:Singleton() {}  // 私有构造函数防止直接实例化Singleton(const Singleton&) = delete;  // 私有拷贝构造函数防止复制Singleton& operator=(const Singleton&) = delete;  // 私有赋值运算符防止赋值static Singleton* instance;  // 静态成员变量存储单例实例public:static Singleton& getInstance() {// 双重检查锁定,确保线程安全if (instance == nullptr) {std::lock_guard<std::mutex> lock(mutex);if (instance == nullptr) {instance = new Singleton();}}return *instance;}private:static std::mutex mutex;  // 互斥锁用于线程安全
};// 在类外初始化静态成员变量
Singleton* Singleton::instance = nullptr;
std::mutex Singleton::mutex;

4.具体代码使用实践
#include <iostream>class Singleton {
public:// Static method to access the singleton instancestatic Singleton& getInstance(){// If the instance doesn't exist, create itif (!instance) {instance = new Singleton();}return *instance;}// Public method to perform some operationvoid someOperation(){std::cout<< "Singleton is performing some operation."<< std::endl;}// Delete the copy constructor and assignment operatorSingleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;private:// Private constructor to prevent external instantiationSingleton(){std::cout << "Singleton instance created."<< std::endl;}// Private destructor to prevent external deletion~Singleton(){std::cout << "Singleton instance destroyed."<< std::endl;}// Private static instance variablestatic Singleton* instance;
};// Initialize the static instance variable to nullptr
Singleton* Singleton::instance = nullptr;int main()
{// Access the Singleton instanceSingleton& singleton = Singleton::getInstance();// Use the Singleton instancesingleton.someOperation();// Attempting to create another instance will not work// Singleton anotherInstance; // This line would not// compilereturn 0;
}

5.注意事项


C++11 实现单例模式的注意事项和潜在问题:

1. 线程安全性:

确保单例类的实现是线程安全的,以避免并行访问时出现数据损坏。使用互斥锁或其他同步机制来保护共享数据。


2. 静态局部变量初始化顺序未定义:

在 C++11 中,静态局部变量(例如 Singleton::instance)的初始化顺序是未定义的。为了确保单例类的正确初始化,请使用双重检查锁定模式。


3. 循环引用和内存泄漏:

如果单例类持有其他对象的引用,请确保这些引用不会导致循环引用并导致内存泄漏。可以使用智能指针或弱引用来管理引用。


4. 异常安全性:

单例类的构造函数应该抛出异常安全,这意味着在构造函数抛出异常后,单例类的状态应该保持有效。可以使用 RAII 技术来实现异常安全性。


5. 过度使用:

避免过度使用单例模式。只在需要全局访问和资源管理的情况下才使用它。过度使用单例模式可能会导致设计僵化和测试困难。

6.最佳实践
  • 使用双重检查锁定:这是一种可靠的方法来确保线程安全的单例实现。
  • 使用 RAII 技术:这可以确保在异常情况下正确清理资源。
  • 仔细管理引用:使用智能指针或弱引用来避免循环引用和内存泄漏。
  • 仅在必要时使用单例模式:考虑其他设计模式,例如工厂模式或依赖注入,以实现全局访问和资源管理。
  • 进行彻底的测试:编写测试用例来验证单例实现的正确性和线程安全性。

7.总结

C++11 中的单例模式通过双重检查锁定和 RAII 技术实现线程安全的惰性实例化,确保全局访问和资源管理。在实现上,这比起低版本的来说,预言特性给我们更加灵活的操作性

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

实验报告5-Spring MVC实现页面

实验报告5-SpringMVC实现页面 一、需求分析 使用Spring MVC框架&#xff0c;从视图、控制器和模型三方面实验动态页面。模拟实现用户登录&#xff0c;模拟的用户名密码以模型属性方式存放在Spring容器中&#xff0c;控制器相应用户请求并映射参数&#xff0c;页面收集用户数据或…

力扣HOT100 - 131. 分割回文串

解题思路&#xff1a; class Solution {List<List<String>> res new ArrayList<>();List<String> pathnew ArrayList<>();public List<List<String>> partition(String s) {backtrack(s,0);return res;}public void backtrack(Str…

C++复盘(一)

文章目录 常量标识符命名规则数据类型sizeof关键字浮点数字符型转义字符字符串型布尔类型bool 比较运算符switch-case语句rand()随机数种子srand() goto语句一维数组函数函数的声明函数的分文件编写 指针指针所占内存空间空指针野指针const修饰指针1、常量指针2、指针常量3、co…

SpringBoot集成Flowable案例

前言 Flowable 是一个使用 Java 编写的轻量级业务流程引擎。Flowable 流程引擎可用于部署 BPMN2.0 流程定义&#xff08;用于定义流程的行业 XML 标准&#xff09;&#xff0c;创建这些流程定义的流程实例&#xff0c;进行查询&#xff0c;访问运行中或历史的流程实例与相关数…

SAP-MM-SD批次管理的影响点M3530

业务场景: 业务部门在创建物料主数据时,勾选了“批次管理”实际不需要。收货时提示输入批次,不能收货了,那回到物料主数据修改,取消勾选“批次管理”发现取消不了,报错M3530,大致内容如下: “显示错误”按钮仅在对话框模式下出现,而不是在数据传输或大规模维护中。 步…

【代码问题】【Pytorch】训练模型时Loss为NaN或INF

解决方法或者问题排查&#xff1a; 加归一化层&#xff1a; 我的问题是我新增的一个模块与原来的模块得到的张量相加&#xff0c;原张量是归一化后的&#xff0c;我的没有&#xff1a; class Module(nn.Module):def __init__(self,dim,):super().__init__()# 新增一个LayerNo…

一、交换网络基础

目录 1.交换机的转发行为 2.数据帧的类型 3.ARP地址解析步骤 Hub&#xff1a;物理层设备 交换机&#xff1a;数据链路层设备 1.交换机的转发行为 泛洪&#xff08;Flooding&#xff09;&#xff08;有可能是单播帧&#xff08;未知单播帧&#xff09;&#xff0c;也有可能是…

有没有一种可能性,你不投递简历,让HR主动联系你

你是否觉得自己得主动给某个公司投递了简历,他们才会联系你,亦或者是自己得主动在招聘APP上联系那个BOSS,他才会反过来跟你说话,又或者是你千方百计的跟他打招呼了,还是没有回应,这一节有可能让你明白,有时候是可以,你不主动,他也会主动联系你的。 目录 1 简历是如何…

webpack3升级webpack4遇到的各种问题汇总

webpack3升级webpack4遇到的各种问题汇总 问题1 var outputNamecompilation.mainTemplate.applyPluginWaterfull(asset-path,outputOptions.filename,{......)TypeError: compilation.mainTemplate.applyPluginsWaterfall is not a function解决方法 html-webpack-plugin 版…

scikit-learn:Python中的机器学习-1

简介&#xff1a;问题设置 什么是机器学习&#xff1f; 机器学习是关于构建具有可调参数的程序&#xff0c;这些参数可以自动调整&#xff0c;以便通过适应先前看到的数据来改善其行为。机器学习可以被认为是人工智能的一个子领域&#xff0c;因为这些算法可以被视为构建模块…

实战—登录功能引发的逻辑漏洞

密码找回功能可能存在的漏洞 1.验证码发送后前端返回 2.验证码无次数限制可爆破 3.验证码可控/邮箱篡改为自己的接收短信验证码/手机号码篡改为自己的接收短信验证码 4.越权漏洞—>自己验证码通过改包然后修改他们密码 5.任意用户密码重置 6.密保问题在前端源码 实战…

【AGX】Ubuntu20.04 + ROS_ noetic+ 大疆Mid360激光 雷达评测

大家好&#xff0c;我是虎哥&#xff0c;最近组装机器人&#xff0c;使用到了大疆孵化的圳市览沃科技有限公司&#xff08;简称Livox览沃科技&#xff09;推出的觅道系列全新混合固态激光雷达Mid-360&#xff0c;顺便试试效果&#xff0c;也记录一下使用入门过程。 "觅道M…