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

「全网最细 + 实战源码案例」设计模式——单例设计模式

核心思想:

  • 属于创建型设计模式,核心目的是确保一个类在整个程序运行期间只有一个实例,并提供一个全局访问点来获取该实例。

  • 控制共享资源的访问(如数据库链接、配置管理、日志处理器等)

  • 真实世界类比:政府是单例模式的一个很好的示例。 一个国家只有一个官方政府。 不管组成政府的每个人的身份是什么,“某政府” 这一称谓总是鉴别那些掌权者的全局访问节点。


结构

所有单例的实现都包含以下两个相同的步骤:

  • 将默认构造函数设为私有,防止其他对象使用单例类的 new 运算符。

  • 新建一个静态构建方法作为构造函数。该函数会“偷偷”调用私有构造函数来创建对象,并将其保存在一个静态成员变量中。此后所有对于该函数的调用都将返回这一缓存对象。

如果你的代码能够访问单例类,那它就能调用单例类的静态方法。无论何时调用该方法,它总是会返回相同的对象。


使用场景:

1. 需要唯一实例的场景:

  • 配置管理类

  • 日志记录器

  • 数据库连接池

  • 多线程环境中的任务调度器

2. 需要全局共享实例

  • 以避免多个实例引发资源冲突或影响程序逻辑。


⭐实现方式:

1. 饿汉式(线程安全,类加载时初始化)

1.1. 静态变量式(常见方式)

// 饿汉式(静态变量)
public class Singleton {// 1. 私有化构造方法private Singleton() {}// 2. 创建一个静态变量,保存实例private static final Singleton instance = new Singleton();// 3. 提供一个公共的静态方法获取实例public static Singleton getInstance() {return instance;}
}

特点:

  • 线程安全:类加载时实例化,JVM 保证线程安全。

  • 缺点:类加载时即创建实例,即使未使用也会占用内存。


1.2. 静态代码块式

// 饿汉式(静态代码块)
public class Singleton {// 1. 私有化构造方法private Singleton(){}// 2. 创建一个静态对象private static Singleton instance;// 3. 在静态代码块中创建对象static {instance = new Singleton();}// 4. 提供获取对象的方法public static Singleton getInstance(){return instance;}
}

特点:

  • 和静态变量方式类似,在类加载时实例化。

  • 可以在静态代码块中加入额外逻辑,例如异常处理或配置初始化。


2. 懒汉式(线程不安全,延迟加载)

// 懒汉式,线程不安全
public class Singleton {// 1. 私有化构造方法private Singleton() {}// 2. 定义一个静态变量,用于存储唯一实例private static Singleton instance;// 3. 定义一个静态方法,用于获取唯一实例public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

优点:

  • 实例在第一次使用时才初始化,节约资源。

缺点:

  • 多线程情况下可能创建多个实例,线程不安全。


3. 线程安全的懒汉式

3.1. 同步方法

// 懒汉式,同步式,线程安全
public class Singleton {// 1. 私有化构造方法private Singleton() {}// 2. 定义一个静态变量,用于存储private static Singleton instance;// 3. 定义一个静态方法,用于获取唯一实例public static synchronized Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

缺点:

  • 同步方法会导致性能下降,尤其是高并发访问时。

3.2. 双重检查锁(推荐)

// 懒汉式,双重检查锁方式
public class Singleton {// 1. 私有化构造方法private Singleton() {}// 2. 定义一个静态变量,用于存储实例,volatile保证可见性与有序性,避免指令重排private static volatile Singleton instance;// 3. 定义一个静态方法,用于获取唯一实例public static Singleton getInstance() {// 1.第一次判断,如果instance的值为null,则进入同步代码块if (instance == null) {// 2.同步代码块,保证线程安全synchronized (Singleton.class) {// 3.第二次判断,如果instance的值为null,则创建实例if (instance == null) {instance = new Singleton();}}}return instance;}
}

优点:

  • 高效,只有在首次实例化时会加锁,之后不会。

注意:volatile 关键字防止指令重排,确保线程安全。

⭐为什么必须要加 volatile

1、防止指令重排

在 Java 中,对象的实例化过程分为三步:

  • 分配内存空间

  • 初始化对象

  • 将内存地址赋值给变量

由于指令重排的存在,步骤 2 和步骤 3 可能被调换执行。例如:

线程 A 在执行 instance = new Singleton() 时,可能执行了分配内存和赋值操作,但还未完成初始化。

此时,instance 已经不为 null,但它指向的对象尚未完全初始化。

如果线程 B 此时调用 getInstance(),判断 instance != null 为真,但实际访问的是一个未初始化完全的对象,这将导致程序出错。

加上 volatile 后,禁止指令重排序,确保初始化顺序正确。

2、保证变量的可见性

Java 的内存模型中,每个线程有自己的工作内存。一个线程对变量的修改,可能不会立即被其他线程所见。

加上 volatile 后,保证每次对 instance 的读操作都能获取到最新的值

当线程 A 完成 instance 初始化后,其他线程(如 B 线程)立刻可见,而不会读取到旧值或中间状态。


3.3. ⭐静态内部类(推荐)

// 懒汉式,静态内部类方式
public class Singleton {// 1.构造函数私有化,外部不能newprivate Singleton() {}// 2.创建静态内部类private static class SingletonHolder {// 3.创建静态变量,保存实例private static final Singleton INSTANCE = new Singleton();}// 3.定义一个静态方法,用于获取唯一实例public static Singleton getInstance() {return SingletonHolder.INSTANCE;}
}

原理:

  • 由于 JVM 加载外部类的过程中,不会加载静态内部类,只有内部类的属性/方法被调用时才会被加载,并初始化其静态属性。静态属性由于被 static 修饰,保证只会被实例化一次,并且严格保证实例化顺序。

优点:

  • 线程安全

  • 实现了延迟加载,按需初始化


4. ⭐枚举单例(最安全,推荐)

// 枚举单例
public enum Singleton {INSTANCE;
}

优点:

  • 简单

  • 天然防止反射和序列化破坏单例


破坏单例

1. 序列化破坏单例

问题:序列化和反序列化可以通过 ObjectInputStream 创建一个新的实例,而不是返回现有的单例实例。

示例代码:

import java.io.*;public class Singleton implements Serializable {private static final long serialVersionUID = 1L;private static final Singleton INSTANCE = new Singleton();private Singleton() {}public static Singleton getInstance() {return INSTANCE;}public static void main(String[] args) throws Exception {Singleton instance1 = Singleton.getInstance();// 将对象序列化到文件ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("singleton.obj"));oos.writeObject(instance1);oos.close();// 从文件反序列化对象ObjectInputStream ois = new ObjectInputStream(new FileInputStream("singleton.obj"));Singleton instance2 = (Singleton) ois.readObject();// 验证是否为同一个实例System.out.println(instance1 == instance2); // 输出:false}
}

原因

  • 序列化机制会通过反序列化的过程创建一个新的对象实例,而不会调用单例类中的 getInstance() 方法。

解决方案:实现 readResolve() 方法,确保反序列化时返回现有实例。

private Object readResolve() {return INSTANCE;
}

2. 反射破坏单例

问题:通过反射,能够直接调用私有构造方法,创建多个实例。

示例代码:

import java.lang.reflect.Constructor;public class Singleton {private static final Singleton INSTANCE = new Singleton();private Singleton() {}public static Singleton getInstance() {return INSTANCE;}public static void main(String[] args) throws Exception {Singleton instance1 = Singleton.getInstance();// 使用反射创建新实例Constructor<Singleton> constructor = Singleton.class.getDeclaredConstructor();constructor.setAccessible(true);Singleton instance2 = constructor.newInstance();// 验证是否为同一个实例System.out.println(instance1 == instance2); // 输出:false}
}

原因

  • 反射可以访问私有构造方法并直接调用,从而绕过单例模式的限制。

解决方案

1、在构造方法中防止重复实例化

private static boolean isCreated = false;private Singleton() {if (isCreated) {throw new RuntimeException("Singleton instance already created!");}isCreated = true;
}

2、使用枚举单例

枚举类的单例天然防止反射和序列化破坏。

public enum Singleton {INSTANCE;
}

3. 总结

  • 序列化破坏:通过 readResolve() 方法解决。

  • 反射破坏:通过构造方法检查或使用枚举单例解决。

  • 推荐方式:使用 枚举单例,最简单且最安全,能有效防止这两种破坏。


在源码中的应用

1. Runtime 类

  • 简介Runtime 类允许应用程序与运行时环境交互,比如调用垃圾回收、运行外部命令等。

  • 实现方式:通过 饿汉式单例 实现。

源码分析:

public class Runtime {private static final Runtime currentRuntime = new Runtime(); // 饿汉式实例化private Runtime() {} // 私有化构造方法public static Runtime getRuntime() {return currentRuntime; // 返回唯一实例}public void gc() {// 调用垃圾回收}public void exit(int status) {// 退出 JVM}
}

特点:
  • 全局唯一实例。

  • 使用饿汉式,保证线程安全。


2. Desktop 类

  • 简介Desktop 类用来打开用户默认的应用程序(如浏览器、邮件客户端等)。

  • 实现方式:通过 懒汉式单例 实现。

源码分析:

public final class Desktop {private static Desktop desktop;private Desktop() {}public static synchronized Desktop getDesktop() {if (desktop == null) {desktop = new Desktop(); // 懒汉式单例}return desktop;}public void browse(URI uri) {// 打开 URI}
}

特点:
  • 使用同步方法保证线程安全。

  • 懒加载,实例在需要时创建。


3. Logger 类( java.util.logging.Logger )

  • 简介Logger 是 Java 的日志工具类,用于记录和管理应用程序日志。

  • 实现方式:内部使用单例模式管理全局日志管理器(LogManager)。

源码分析(核心部分):

public class Logger {private static final LogManager manager = LogManager.getLogManager(); // 单例的 LogManagerprotected Logger(String name, String resourceBundleName) {// Logger 构造方法}public static Logger getLogger(String name) {return manager.getLogger(name); // 通过单例 LogManager 获取 Logger}
}

特点:

  • LogManager 作为单例管理所有 Logger 实例。

  • getLogger 方法确保每个名称对应的 Logger 是唯一的。


4. 总结

在 JDK 源码中,单例模式被广泛应用于需要 全局唯一实例 或 资源共享 的场景:

  1. 饿汉式Runtime 类。

  2. 懒汉式Desktop 类。

  3. 组合模式Logger 类中的 LogManager 单例。

这些设计的核心目标是:确保全局状态的一致性、节省资源以及简化管理

单例模式优缺点:

与其他模式的关系:

  1. 外观模式类通常可以转换为单例模式类, 因为在大部分情况下一个外观对象就足够了。

  2. 如果你能将对象的所有共享状态简化为一个享元对象, 那么享元模式就和单例类似了。 但这两个模式有两个根本性的不同。

  3. 只会有一个单例实体, 但是享元类可以有多个实体, 各实体的内在状态也可以不同。

    单例对象可以是可变的。 享元对象是不可变的。

  4. 抽象工厂模式、 生成器模式原型模式都可以用单例来实现。

文章转载自:SlackClimb

原文链接:「全网最细 + 实战源码案例」设计模式——单例设计模式 - SlackClimb - 博客园

体验地址:引迈 - JNPF快速开发平台_低代码开发平台_零代码开发平台_流程设计器_表单引擎_工作流引擎_软件架构

相关文章:

「全网最细 + 实战源码案例」设计模式——单例设计模式

核心思想&#xff1a; 属于创建型设计模式&#xff0c;核心目的是确保一个类在整个程序运行期间只有一个实例&#xff0c;并提供一个全局访问点来获取该实例。 控制共享资源的访问&#xff08;如数据库链接、配置管理、日志处理器等&#xff09; 真实世界类比&#xff1a;政府…...

第01章 19 通过点数据逐级构建球体体数据的综合性小例子

用VTK库来创建一个三维图像数据&#xff08;vtkImageData&#xff09;&#xff0c;并填充标量数据以表示一个球体的体数据。球的半径为50&#xff0c;体数据的空间间隔为1.0/1000。 首先&#xff0c;我需要包含VTK的头文件&#xff0c;并且创建一个vtkImageData对象。然后&…...

CVE-2024-23897-Jenkins任意文件读取漏洞复现

content Jenkins是什么CVE-2024-23897总结修复建议 Jenkins是什么 Jenkins是一人基于Java开发的、可扩展的持续集成引擎&#xff0c;用于持续、自动地构建/测试软件项目&#xff0c;可以监控一些定时执行的任务。 官网文档&#xff1a; Jenkins是一款开源 CI&CD 软件&…...

前端react后端java实现提交antd form表单成功即导出压缩包

前端&#xff08;React Ant Design&#xff09; 1. 创建表单&#xff1a;使用<Form>组件来创建你的表单。 2. 处理表单提交&#xff1a;在onFinish回调中发起请求到后端API&#xff0c;并处理响应。 import React from react; import { Form, Input, Button } from ant…...

基于ESP32的桌面小屏幕实战[6]:环境搭建和软件基础

摘要 本文分为两部分&#xff1a;Linux开发环境搭建和软件基础。Linux开发环境搭建介绍了Ubuntu虚拟机安装及SSH、Samba配置&#xff0c;可以实现用VSCode操作虚拟机。为了后续工作&#xff0c;搭建了乐鑫ESP32 SDK环境。软件基础介绍了Linux开发常用的软件基础&#xff0c;包…...

接口(完)

大家好&#xff0c;今天我们着重来总结一下接口的知识&#xff0c;并且将接口和抽象类的区别罗列一下&#xff0c;帮助我们更好的认识抽象类和接口。 2.7 抽象类和接口的区别. 抽类和接口都是Java中多态的常见使用方式,都需要重点掌握,同时又要认清两者的区别(重要!!&#xf…...

数据结构——实验七·排序

欢迎各位大佬们来到Tubishu的博客&#x1f31f; Tubishu是一名计算机本科生&#xff0c;不定期发送一些在学校的成果供佬们消遣~希望能为佬的编程之路添砖加瓦⭐&#x1f525; 求各位大佬们垂怜&#x1f525;点赞评论一下呗&#x1f525;&#x1f525; 本文专栏 ➡️ 数据结构 …...

JVM堆空间

JVM&#xff08;Java虚拟机&#xff09;堆空间是Java内存管理的核心区域之一&#xff0c;用于存储Java对象实例。以下是关于JVM堆空间的详细介绍&#xff1a; 1. 堆空间的作用 • 存储对象实例&#xff1a;几乎所有的Java对象实例&#xff08;通过new关键字创建的对象&#xf…...

【详细】SSH公私钥认证与渗透测试攻击场景

SSH&#xff08;Secure Shell&#xff09;是一个用于远程登录和执行命令的网络协议&#xff0c;其认证方式通常有两种&#xff1a;基于密码的认证和基于公私钥的认证。本文将详细介绍SSH公私钥认证机制&#xff0c;并探讨在渗透测试场景中&#xff0c;如何利用对靶机具有读取和…...

常见的多媒体框架(FFmpeg GStreamer DirectShow AVFoundation OpenMax)

1.FFmpeg FFmpeg是一个非常强大的开源多媒体处理框架&#xff0c;它提供了一系列用于处理音频、视频和多媒体流的工具和库。它也是最流行且应用最广泛的框架&#xff01; 官方网址&#xff1a;https://ffmpeg.org/ FFmpeg 的主要特点和功能&#xff1a; 编解码器支持: FFmpe…...

C++异步future

&#x1f30e; C11异步futrue 文章目录&#xff1a; C11异步futrue future介绍     应用场景     future操作       std::async函数模版       std::packaged_task类模版       std::promise类模版 &#x1f680;future介绍 std::future是C11标准库…...

Oracle 12c 中的 CDB和PDB的启动和关闭

一、简介 Oracle 12c引入了多租户架构&#xff0c;允许一个容器数据库&#xff08;Container Database, CDB&#xff09;托管多个独立的可插拔数据库&#xff08;Pluggable Database, PDB&#xff09;。本文档旨在详细描述如何启动和关闭CDB及PDB。 二、容器数据库 (CDB) 2.1…...

Vue组件开发-使用 html2canvas 和 jspdf 库实现PDF文件导出 设置页面大小及方向

在 Vue 项目中实现导出 PDF 文件、调整文件页面大小和页面方向的功能&#xff0c;使用 html2canvas 将 HTML 内容转换为图片&#xff0c;再使用 jspdf 把图片添加到 PDF 文件中。以下是详细的实现步骤和代码示例&#xff1a; 步骤 1&#xff1a;安装依赖 首先&#xff0c;在项…...

chrome插件:网页图片高清下载

前置条件&#xff1a; 安装有chrome谷歌浏览器的电脑 使用步骤&#xff1a; 1.打开chrome扩展插件 2.点击管理扩展程序 3.加载已解压的扩展程序 4.选择对应文件夹 5.成功后会出现一个扩展小程序 6.点击对应小程序 7.输入需要访问的网址&#xff0c;点击扩展插件即可进行图片…...

汽车定速巡航

配备定速巡航功能的车型&#xff0c;一般在方向盘附近设有4~6个按键&#xff08;可能共用键位&#xff09;。 要设置定速巡航&#xff0c;不仅需要方向盘上的按键&#xff0c;还要油门配合。 设置的一般流程&#xff1a; 开关&#xff1a;类似步枪上的“保险”&#xff0c;按…...

CNN-BiLSTM卷积双向长短期记忆神经网络时间序列预测(Matlab完整源码和数据)

CNN-BiLSTM卷积双向长短期记忆神经网络时间序列预测&#xff08;Matlab完整源码和数据&#xff09; 目录 CNN-BiLSTM卷积双向长短期记忆神经网络时间序列预测&#xff08;Matlab完整源码和数据&#xff09;预测效果基本介绍 CNN-BiLSTM卷积双向长短期记忆神经网络时间序列预测一…...

WPF基础 | WPF 布局系统深度剖析:从 Grid 到 StackPanel

WPF基础 | WPF 布局系统深度剖析&#xff1a;从 Grid 到 StackPanel 一、前言二、Grid 布局&#xff1a;万能的布局王者2.1 Grid 布局基础&#xff1a;构建网格世界2.2 子元素定位与跨行列&#xff1a;布局的精细操控2.3 自适应布局&#xff1a;灵活应变的秘诀 三、StackPanel…...

14-6-2C++STL的list

(一&#xff09;list对象的带参数构造 1.list&#xff08;elem);//构造函数将n个elem拷贝给本身 #include <iostream> #include <list> using namespace std; int main() { list<int> lst(3,7); list<int>::iterator it; for(itlst.begi…...

【AI论文】Sigma:对查询、键和值进行差分缩放,以实现高效语言模型

摘要&#xff1a;我们推出了Sigma&#xff0c;这是一个专为系统领域设计的高效大型语言模型&#xff0c;其独特之处在于采用了包括DiffQKV注意力机制在内的新型架构&#xff0c;并在我们精心收集的系统领域数据上进行了预训练。DiffQKV注意力机制通过根据查询&#xff08;Q&…...

私域流量池构建与转化策略:以开源链动2+1模式AI智能名片S2B2C商城小程序为例

摘要&#xff1a;随着互联网技术的快速发展&#xff0c;流量已成为企业竞争的关键资源。私域流量池&#xff0c;作为提升用户转化率和增强用户粘性的有效手段&#xff0c;正逐渐受到企业的广泛关注。本文旨在深入探讨私域流量池构建的目的、优势及其在实际应用中的策略&#xf…...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)

0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述&#xff0c;后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作&#xff0c;其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

django filter 统计数量 按属性去重

在Django中&#xff0c;如果你想要根据某个属性对查询集进行去重并统计数量&#xff0c;你可以使用values()方法配合annotate()方法来实现。这里有两种常见的方法来完成这个需求&#xff1a; 方法1&#xff1a;使用annotate()和Count 假设你有一个模型Item&#xff0c;并且你想…...

渲染学进阶内容——模型

最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...

P3 QT项目----记事本(3.8)

3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...

Axios请求超时重发机制

Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式&#xff1a; 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

聊一聊接口测试的意义有哪些?

目录 一、隔离性 & 早期测试 二、保障系统集成质量 三、验证业务逻辑的核心层 四、提升测试效率与覆盖度 五、系统稳定性的守护者 六、驱动团队协作与契约管理 七、性能与扩展性的前置评估 八、持续交付的核心支撑 接口测试的意义可以从四个维度展开&#xff0c;首…...

VM虚拟机网络配置(ubuntu24桥接模式):配置静态IP

编辑-虚拟网络编辑器-更改设置 选择桥接模式&#xff0c;然后找到相应的网卡&#xff08;可以查看自己本机的网络连接&#xff09; windows连接的网络点击查看属性 编辑虚拟机设置更改网络配置&#xff0c;选择刚才配置的桥接模式 静态ip设置&#xff1a; 我用的ubuntu24桌…...

Razor编程中@Html的方法使用大全

文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...

打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用

一、方案背景​ 在现代生产与生活场景中&#xff0c;如工厂高危作业区、医院手术室、公共场景等&#xff0c;人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式&#xff0c;存在效率低、覆盖面不足、判断主观性强等问题&#xff0c;难以满足对人员打手机行为精…...

Proxmox Mail Gateway安装指南:从零开始配置高效邮件过滤系统

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storms…...