设计模式-创建型模式之单例模式

单例模式属于创建型模式,一个单例类在任何情况下都只存在一个实例,构造方法必须是私有的、由自己创建一个静态变量存储实例,对外提供个静态公有方法获取实例。

单例模式的主要角色:

  • 单例类:只能创建一个实例的类
  • 访问类:使用单例类

单例设计模式分为两种:

  • 饿汉式:类加载就会导致该单实例对象被创建
  • 懒汉式:类加载不会导致该单实例对象被创建,而是首次使用该对象时才会被创建

优点:内存中只有一个实例,不会频繁创建和销毁对象,减少了开销,节省系统资源。

缺点:没有抽象层,难以扩展,与单一职责原则冲突。

单例模式常见的写法有哪些?(手写常见单例模式)

(1)饿汉式:线程安全。

**优点:**线程安全,没有加锁,执行效率较高

**缺点:**不是懒加载,类加载时就初始化,浪费内存空间

**(2)懒汉式(线程不安全):**当第一次调用getInstance()方法获取Singleton类的对象的时候才创建Singleton类的对象,这样就实现了懒加载的效果。但是,如果是多线程环境,会出现线程安全问题。

**(3)懒汉式(线程安全):**使用 synchronized 关键字确保线程安全,但是每⼀次调⽤ getInstance 获取实例时都需要加锁和释放锁,影响性能,效率低下。

**(4)双重检查锁(DCL,即double-checked locking):**对于 `getInstance()` 方法来说,绝大部分的操作都是读操作,读操作是线程安全的,所以没必要每个线程必须持有锁才能调用该方法,我们需要调整加锁的时机。

**优点:**懒加载,线程安全,效率较高

**缺点:**实现较复杂

Volatile关键字可以保证`INSTANCE`在线程之间的可见性和有序性

添加 `volatile` 关键字之后的双重检查锁模式是一种比较好的单例实现模式,能够保证在多线程的情况下线程安全也不会有性能问题。

**(5)类静态内部:**静态内部类单例模式中实例由内部类创建,由于 JVM 在加载外部类的过程中, 是不会加载静态内部类的, 只有内部类的属性/方法被调用时才会被加载, 并初始化其静态属性。静态属性由于被 `static` 修饰,保证只被实例化一次,并且严格保证实例化顺序。不用加锁懒加载,线程安全,实现简单,效率较高

第一次加载Singleton类时不会去初始化INSTANCE,只有第一次调用getInstance,虚拟机加载Holder,并初始化INSTANCE,这样不仅能确保线程安全,也能保证 Singleton 类的唯一性。

​静态内部类单例模式是一种优秀的单例模式,是开源项目中比较常用的一种单例模式。在没有加任何锁的情况下,保证了多线程下的安全,并且没有任何性能影响和空间的浪费。

相关推荐
wuminyu1 小时前
Java锁机制之park与futex系统级协同机制解析
java·linux·c语言·jvm·c++
疯狂打码的少年1 小时前
编译程序与解释程序的区别
java·开发语言·笔记
xieliyu.8 小时前
Java算法精讲:双指针(三)
java·开发语言·算法
明夜之约8 小时前
Spring Boot 自动装配源码
java·spring boot·后端
Leaton Lee8 小时前
Spring Boot分层架构详解:从Controller到Service再到Mapper的完整流程
java·spring boot·后端·架构
Jinkxs8 小时前
Resilience4j- 与 Spring Boot 快速集成:自动配置与基础注解使用
java·spring boot·后端
辣机小司8 小时前
【踩坑记录:Spring Boot 配置文件读取值不一致?警惕 YAML 的“八进制陷阱”与 SnakeYAML 版本之谜】
java·spring boot·后端·yaml·踩坑记录
fangdengfu1239 小时前
ES分析系统各个服务日志占用量
java·前端·elasticsearch
云烟成雨TD9 小时前
Spring AI 1.x 系列【51】可观测性技术选型
java·人工智能·spring
星越华夏10 小时前
ESP32-CAM图像传输项目说明文档
java·后端·struts·esp32