【设计模式】【创建型模式(Creational Patterns)】之单例模式

server/2024/11/24 15:00:32/

单例模式是一种常用的创建型设计模式,其目的是确保一个类只有一个实例,并提供一个全局访问点。

单例模式的原理

单例模式的核心在于控制类的实例化过程,通常通过以下方式实现:

  1. 私有化构造函数,防止外部直接实例化。
  2. 提供一个静态方法或属性,用于返回类的唯一实例。
  3. 使用懒汉式或饿汉式初始化策略来创建实例。

UML 类图

下面是单例模式的UML类图:

+-----------------+
| Singleton       |
+-----------------+
| -instance: Singleton |
+-----------------+
| +getInstance(): Singleton |
| +doSomething(): void    |
+-----------------+

在这个UML类图中:

  • Singleton 类有一个私有的静态变量 instance,用来存储唯一的实例。
  • getInstance() 方法是一个静态方法,用于获取 Singleton 类的唯一实例。
  • doSomething() 是一个普通的方法,用于演示单例对象的功能。

Java 代码示例

这里提供两种常见的单例模式实现方式:懒汉式和饿汉式。

饿汉式(线程安全,加载速度较慢)
public class Singleton {// 在静态初始化器中创建实例,保证线程安全private static final Singleton INSTANCE = new Singleton();// 私有构造函数,防止外部实例化private Singleton() {}// 公共静态方法,提供全局访问点public static Singleton getInstance() {return INSTANCE;}// 示例方法public void doSomething() {System.out.println("Doing something...");}
}
懒汉式(线程安全,延迟加载)
public class Singleton {// 私有静态变量,初始值为 nullprivate static Singleton instance;// 私有构造函数,防止外部实例化private Singleton() {}// 公共静态方法,提供全局访问点public static synchronized Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}// 示例方法public void doSomething() {System.out.println("Doing something...");}
}

懒汉式优化:双重检查锁定(Double-Checked Locking)

为了提高性能,可以使用双重检查锁定来减少同步锁的开销:

public class Singleton {// 使用 volatile 关键字确保多线程环境下的可见性private static volatile Singleton instance;// 私有构造函数,防止外部实例化private Singleton() {}// 公共静态方法,提供全局访问点public static Singleton getInstance() {if (instance == null) {synchronized (Singleton.class) {if (instance == null) {instance = new Singleton();}}}return instance;}// 示例方法public void doSomething() {System.out.println("Doing something...");}
}

C++ 实现

在 C++ 中,我们可以使用静态成员变量和静态成员函数来实现单例模式

#include <iostream>class Singleton {
private:// 私有构造函数,防止外部实例化Singleton() {}// 删除拷贝构造函数和赋值操作符,防止拷贝Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;// 静态成员变量,存储唯一的实例static Singleton* instance;public:// 静态成员函数,提供全局访问点static Singleton* getInstance() {if (instance == nullptr) {instance = new Singleton();}return instance;}// 示例方法void doSomething() {std::cout << "Doing something..." << std::endl;}
};// 初始化静态成员变量
Singleton* Singleton::instance = nullptr;int main() {Singleton* singleton = Singleton::getInstance();singleton->doSomething();return 0;
}

Python 实现

在 Python 中,可以通过模块级别的全局变量来实现单例模式,也可以使用装饰器或者元类。

class Singleton:_instance = Nonedef __new__(cls):if not cls._instance:cls._instance = super(Singleton, cls).__new__(cls)return cls._instancedef do_something(self):print("Doing something...")# 测试
singleton1 = Singleton()
singleton2 = Singleton()print(singleton1 is singleton2)  # 输出 True,表明它们是同一个实例
singleton1.do_something()

Go 实现

在 Go 中,可以使用包级变量来实现单例模式

package mainimport "fmt"type Singleton struct{}var instance *Singletonfunc GetInstance() *Singleton {if instance == nil {instance = &Singleton{}}return instance
}func (s *Singleton) DoSomething() {fmt.Println("Doing something...")
}func main() {singleton := GetInstance()singleton.DoSomething()
}

解释

  • C++:使用了静态成员变量和静态成员函数来确保单例的唯一性和全局可访问性。同时删除了拷贝构造函数和赋值操作符,防止拷贝。
  • Python:使用了类的 __new__ 方法来控制实例的创建,确保只有一个实例存在。
  • Go:使用了包级变量 instance 来存储唯一的实例,并通过 GetInstance 函数来提供全局访问点。

总结

这些实现都遵循了单例模式的基本原则:确保一个类只有一个实例,并提供一个全局访问点。每种语言的实现方式有所不同,但核心思想是一致的。


http://www.ppmy.cn/server/144569.html

相关文章

以3D数字人AI产品赋能教育培训人才发展,魔珐科技亮相AI+教育创新与人才发展大会

11月20日&#xff0c;北京中关村国际创新中心迎来了“AI教育创新与人才发展大会暨首届北京数字人才发展大会”的盛大启幕。此次大会汇聚了培训、教育、科技、人才领域的专家学者、行业领袖及企业代表&#xff0c;共同探讨人工智能技术在教育培训领域的革新应用与数字人才培养体…

C语言练级->##__VA_ARGS__(可变参数)的用法

有什么用&#xff1f; 通常__VA_ARGS__用于宏定义&#xff0c;其中关于日志宏需要用的&#xff0c;printf 等支持可变参数的函数的宏封装。 首先我们先知道这个__VA_ARGS__的英文全称是“Variadic Arguments” 叫可变参数。说到可变参数学过C语言的朋友们应该都会想到printf&…

“LLM是否是泡沫”

目录 “LLM是否是泡沫” 培养自己鉴别论文价值的能力、复现开源项目的能力、debug 代码的能力 llm 是生产力工具 多去找实习&#xff0c;读再多的论文&#xff0c;刷再多的技术文章&#xff0c;也不如一次 debug 多机通讯报错带来的认知深刻 一、LLM领域的发展与挑战 二、…

【LeetCode每日一题】——746.使用最小花费爬楼梯

文章目录 一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【题目提示】七【解题思路】八【时空频度】九【代码实现】十【提交结果】 一【题目类别】 数组 二【题目难度】 简单 三【题目编号】 746.使用最小花费爬楼梯 四【题目描述】 给你一…

node读取execl或写入execl数据保存

nodejs 使用 exceljs 库读取 execl 或写入 execl 数据后保存文件 安装库 exceljs npm i exceljs 读取execl const exceljs require(exceljs)const workbook new exceljs.Workbook() await workbook.xlsx.readFile(test.xlsx) // 读取第一个工作表 const worksheet workbo…

使用docker compose安装部署gitlab

安装gitlab docker pull gitlab/gitlab-ce:latest下载并安装 Docker Compose V2&#xff1a; sudo curl -L "https://github.com/docker/compose/releases/download/v2.17.2/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose sudo chmod…

基于Java Springboot高校教室资源管理系统

一、作品包含 源码数据库全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 数据库&#xff1a;…

【RK3588 Linux 5.x 内核编程】-内核中断与Tasklet

内核中断与Tasklet 文章目录 内核中断与Tasklet1、Tasklet介绍2、创建Tasklet2.1 创建Tasklet2.2 动态方式创建Tasklet3、启用和禁用Tasklet4、Tasklet调度5、杀掉Tasklet6、Tasklet使用示例7、驱动验证在前面的文章中,对Linux的内核中断做了详细的介绍。我们知道,在Linux内核…