【创建型设计模式】单例模式

【创建型设计模式】单例模式

这篇博客接下来几篇都将阐述设计模式相关内容。

接下来的顺序大概是:单例模式、工厂方法模式、抽象工厂模式、建造者模式、原型模式。

一、什么是单例模式

单例模式是一种创建型设计模式,它保证一个类仅有一个实例,并提供一个全局访问点。

核心思想:

  1. 限制类的实例化次数,确保全局只有一个实例。
  2. 提供统一访问该实例的方法。

Client 为客户端,Singleton 是单例类,通过调用 Singleton.getInstance() 来获取实例对象。

二、单例模式的 6 种方法

(1) 饿汉模式
public class Singleton {private static Singleton instance = new Singleton();private Singleton() {}public static Singleton getInstance() {return instance;}
}

特点:

  • 初始化时间: 类加载时即完成实例化。
  • 访问效率: 调用 getInstance 时无需等待,性能高。

优点:

  • 基于类加载机制,天然线程安全。
  • 实现简单,适用于单例对象较少被频繁初始化的场景。

缺点:

  • 无法延迟加载,如果实例从未被使用,会浪费内存。
(2) 懒汉模式(线程不安全)
public class Singleton {private static Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

特点:

  • 初始化时间: 第一次调用时才实例化。
  • 访问效率: 多线程环境下存在问题,可能生成多个实例。

优点: 延迟加载,节省资源。

缺点: 线程不安全,多个线程同时调用可能创建多个实例,导致单例失效。

(3) 懒汉模式(线程安全)
public class Singleton {private static Singleton instance;private Singleton() {}public static synchronized Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

特点:

  • 线程安全性: 使用 synchronized 关键字确保多线程下只生成一个实例。

优点: 在多线程环境中保证安全。

缺点: 同步锁导致性能下降,每次调用 getInstance 都需排队,影响效率。

(4) 双重检查模式(DCL)
public class Singleton {// 1. 声明 volatile 修饰的静态实例变量private static volatile Singleton instance;// 2. 私有化构造方法,防止外部直接实例化private Singleton() {}// 3. 提供对外获取实例的静态方法public static Singleton getInstance() {// 第一次检查:避免不必要的同步if (instance == null) {synchronized (Singleton.class) { // 加锁// 第二次检查:确保实例未被其他线程创建if (instance == null) {instance = new Singleton();}}}return instance;}
}

特点:

  • 性能优化: 减少同步开销。
  • 线程安全: 使用 volatile 避免指令重排序,确保多线程环境下正确创建实例。

优点: 资源利用率高,延迟初始化,线程安全,性能优于懒汉模式(线程安全版)。

缺点: 实现复杂,某些情况下可能出现 DCL 失效问题(如旧版 JVM)。

DCL 模式的核心思想

DCL 的核心是两次检查同步锁的结合。

**第一次检查(if (instance == null)):**在大多数情况下,实例已经被初始化,可以直接返回,不需要加锁。减少了不必要的同步,提高性能。

**加锁(synchronized):**在实例尚未初始化时,进入临界区,防止多个线程同时创建实例。

**第二次检查(if (instance == null)):**加锁后,再次检查实例是否为 null,防止多个线程同时通过第一次检查,确保单例对象只被创建一次。

**volatile 关键字:**确保 instance 的修改对所有线程立即可见,防止指令重排序导致未完全初始化的对象被其他线程访问。

为什么需要 volatile

问题:指令重排序

在没有 volatile 修饰时,JVM 编译器和 CPU 可能对以下代码进行优化:

instance = new Singleton();

这一行代码可能被分解为以下三步:

  1. 为对象分配内存。
  2. 初始化对象。
  3. 将对象引用赋值给 instance

但实际执行中,可能发生指令重排序

  1. 为对象分配内存。
  2. 将对象引用赋值给 instance
  3. 初始化对象。

如果线程 A 在步骤 2 后被切换,线程 B 进入并访问了 instance,此时会得到一个未初始化的对象,导致程序行为不可预期。

解决方法:volatile 关键字禁止指令重排序,确保对象初始化完成后才将引用赋值给 instance

(5) 静态内部类单例模式
public class Singleton {// 私有化构造方法,防止外部实例化private Singleton() {}// 静态内部类,持有单例实例private static class SingletonHolder {private static final Singleton INSTANCE = new Singleton();}// 提供全局访问点,返回静态内部类中的单例实例public static Singleton getInstance() {return SingletonHolder.INSTANCE;}
}

特点:

  • 延迟加载: 第一次调用 getInstance 时才加载内部类,完成实例化。
  • 线程安全: 内部类加载机制天然线程安全。

优点:

  • 实现简单,避免同步开销。
  • 兼具延迟加载和线程安全,推荐使用。

缺点: 无法控制实例销毁,适用单例生命周期较长的场景。

静态内部类的加载时机:

在 Java 中,类的加载是延迟的,只有在被真正使用时才会加载静态内部类。

主类被加载时,静态内部类不会被加载。

只有在调用静态内部类中的成员时,静态内部类才会被加载。

类加载的线程安全性:

JVM 在加载类时会自动保证线程安全。

静态变量 INSTANCE 只会被初始化一次。

(6) 枚举单例
public enum Singleton {INSTANCE; // 枚举单例的唯一实例// 单例类的其他方法public void doSomething() {System.out.println("Singleton instance is working!");}
}

特点:

  • 线程安全: 枚举的线程安全由 JVM 保证。
  • 防反序列化破坏: 默认枚举实例不可反序列化。

优点:

  • 实现简单,天生线程安全。
  • 避免反序列化和反射破坏单例。

缺点:

  • 不支持延迟加载。
  • 某些场景下(如需要灵活实例化控制)不适用。
public class Test {public static void main(String[] args) {Singleton instance = Singleton.INSTANCE;instance.doSomething();}
}

为什么枚举单例是线程安全的?

**JVM 保证:**枚举类型在类加载时会由 JVM 初始化,类加载过程是线程安全的。

**枚举的单一实例特性:**枚举中的每个实例在加载时被自动创建,并且不可被外部修改。

三、单例实现对比

实现方式是否线程安全是否延迟加载性能复杂度
饿汉模式简单
懒汉模式(不安全)简单
懒汉模式(同步)低(频繁加锁)简单
双重检查模式(DCL)较高中等
静态内部类模式简单
枚举单例简单

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

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

相关文章

【Elasticsearch入门到落地】2、正向索引和倒排索引

接上篇《1、初识Elasticsearch》 上一篇我们学习了什么是Elasticsearch,以及Elastic stack(ELK)技术栈介绍。本篇我们来什么是正向索引和倒排索引,这是了解Elasticsearch底层架构的核心。 上一篇我们学习到,Elasticsearch的底层是由Lucene实…

【Spring Boot】# 使用@Scheduled注解无法执行定时任务

1. 前言 在 Spring Boot中,使用Scheduled注解来定义定时任务时,定时任务不执行;或未在规定时间执行。 import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component;Component public c…

STM32总体架构简单介绍

目录 一、引言 二、STM32的总体架构 1、三个被动单元 (1)内部SRAM (2)内部闪存存储器 (3)AHB到APB的桥(AHB to APBx) 2、四个主动(驱动)单元 &#x…

C# Postman或者PostApi调试前端webapi接口发送带有request/body/head信息

知识: 前端接口,表单形式提交。 req.ContentType "application/x-www-form-urlencoded"; x-www-form-urlencoded 是一种常见的 MIME 类型,用于将键值对编码为 HTTP 请求体中的 URL 编码格式。在 Web API 中,x-www-for…

李宏毅机器学习课程知识点摘要(1-5集)

前5集 过拟合: 参数太多,导致把数据集刻画的太完整。而一旦测试集和数据集的关联不大,那么预测效果还不如模糊一点的模型 所以找的数据集的量以及准确性也会影响 由于线性函数的拟合一般般,所以用一组函数去分段来拟合 sigmoi…

七、SElinux

一、SElinux简介 SELinux是Security-Enhanced Linux的缩写,意思是安全强化的linuxSELinux 主要由美国国家安全局(NSA)开发,当初开发的目的是为了避免资源的误用传统的访问控制在我们开启权限后,系统进程可以直接访问当我们对权限设置不严谨时…

小程序25- iconfont 字体图标的使用

项目中使用到图标,一般由公司设计进行设计,设计好后上传到阿里巴巴矢量图标库 日常开发过程中,也可以通过 iconfont 图标库下载使用自带的图标 补充:使用 iconfont 图标库报错:Failed to load font 操作步骤&#xff…

鸢尾花植物的结构认识和Python中scikit-learn工具包的安装

鸢尾花植物的结构认识和Python中scikit-learn工具包的安装 鸢尾花植物的结构认识和Python中scikit-learn工具包的安装 鸢尾花植物的结构认识和Python中scikit-learn工具包的安装一、鸢尾花的认识1.1 对花结构和功能认识1.2、鸢尾花认识1.2.1 鸢尾花种类1.2.2 鸢尾花结构 二. Py…

Unity3D 截图

使用 Unity3D 自带的截图接口,制作截图工具。 截图 有时候我们想对 Unity 的窗口进行截图,如果直接使用一些截图工具,很难截取到一张完整分辨率的图片(例如,我们想要截取一张 1920 * 1080 的图片)。 其实…

Mysql的加锁情况详解

最近在复习mysql的知识点,像索引、优化、主从复制这些很容易就激活了脑海里尘封的知识,但是在mysql锁的这一块真的是忘的一干二净,一点映像都没有,感觉也有点太难理解了,但是还是想把这块给啃下来,于是想通…

丹摩征文活动 | AI创新之路,DAMODEL助你一臂之力GPU

目录 前言—— DAMODEL(丹摩智算) 算力服务 直观的感受算力提供商的强大​ 平台功能介绍​ 镜像选择 云磁盘创建 总结 前言—— 只需轻点鼠标,开发者便可拥有属于自己的AI计算王国 - 从丰富的GPU实例选择,到高性能的云磁盘,再到预配置的深度学习…

Linux之日志

日志 在编写网络服务器, 各种软件时, 程序一定要打印一些日志信息. 1. 可以向显示器打印, 也可以向文件中写入. 2. 日志是软件在运行时记录的流水账, 用于排查服务进程挂掉的信息. 其中必须要有的是: 日志等级, 时间, 日志内容.可选的是文件名, 代码行数, 进程pid 等 日志…

IDEA指定Maven的settings不生效问题处理

文章目录 一、问题描述二、问题分析三、问题解决 一、问题描述 在Idea中手动指定了maven的settings配置文件,但是一直没生效。 如下图:设置加载settings-aliyun.xml文件,但是最后发现还是在加载settings.xml文件 二、问题分析 ‌在Intel…

【软考】数据库

1. 数据模型 1.1 概念数据模型 概念数据模型一般用 E-R 图表示,常用术语如下: 实体:客观存在的事物,如:一个单位、一个职工、一个部门、一个项目。属性:学生实体有学号、姓名、出生日期等属性。码&#…

oneplus6线刷、trwp、magisk(apatch)、LSPosed、Shamiko、Hide My Applist

oneplus6线刷android10.0.1 oneplus6线刷包(官方android10.0.1)下载、线刷教程: OnePlus6-brick-enchilada_22_K_52_210716_repack-HOS-10_0_11-zip 启用开发者模式 设置 / 连续点击6次版本号 : 启用开发者模式设置/开发者模式/{打开 usb调试, 打开 网络adb调试,…

ByteBuffer模拟拆包输出消息字符串

以下代码模拟网络编程中的粘包现象,用\n进行分割消息块 源码 public static void main(String[] args) {ByteBuffer byteBuffer1 ByteBuffer.allocate(60) ;byteBuffer1.put("Hello World\nWhat is you name?\nI am Licky!\nHo".getBytes());splice(byt…

成都睿明智科技有限公司怎么样可靠不?

在这个日新月异的数字时代,电商行业如同一股不可阻挡的洪流,席卷着每一个消费者的生活。而抖音,作为短视频与电商完美融合的典范,更是为无数商家开辟了一片全新的蓝海。在这片充满机遇与挑战的海洋中,成都睿明智科技有…

【计算机网络】多路转接之epoll

epoll也是一种linux中的多路转接方案(epoll也是只负责IO过程中的"等") 一、epoll相关接口的使用 1.epoll_create int epoll_create(int size); ​功能:创建一个epoll模型 ① int size:没意义了 >0就行 返回值:返回一个文件…

Linux高阶——1117—TCP客户端服务端

目录 1、sock.h socket常用函数 网络初始化函数 首次响应函数 测试IO处理函数 获取时间函数 总代码 2、sock.c SOCKET() ACCEPT()——服务端使用这个函数等待客户端连接 CONNECT()——客户端使用这个函数连接服务端 BIND()——一般只有服务端使用 LISTEN()——服务端…

【SVN和GIT】版本控制系统详细下载使用教程

文章目录 ** 参考文章一、什么是SVN和GIT二、软件使用介绍1 SVN安装1.1 服务端SVN下载地址1.2 客户端SVN下载地址2 SVN使用2.1 服务端SVN基础使用2.1.1 创建存储库和用户成员2.1.2 为存储库添加访问人员2.2 客户端SVN基础使用2.2.1 在本地下载库中的内容2.2.2 版本文件操作--更…