单例模式的理解和实践

        在软件开发中,设计模式是开发者在特定情境下,对常见问题的通用解决方案。这些模式帮助开发者以更高效、可维护的方式编写代码。其中,单例模式(Singleton Pattern)是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将详细讲解单例模式的理解及其在Java中的实践。

一、单例模式的理解

1.1 定义

单例模式是一种确保某个类只有一个实例,并提供一个全局访问点的设计模式。它适用于以下场景:

  • 资源控制:如数据库连接池、文件系统等,需要控制资源的使用,避免频繁创建和销毁对象。
  • 共享状态:如配置管理器、线程池等,需要在多个地方共享同一个状态。

1.2 关键点

  • 私有构造函数:防止外部通过new关键字创建实例。
  • 静态实例:在类内部维护一个静态实例。
  • 全局访问点:提供一个静态方法,用于返回该实例。

1.3 实现步骤

  1. 私有化构造函数:防止外部通过构造函数创建实例。
  2. 创建静态实例:在类内部定义一个静态变量,并初始化为类的实例。
  3. 提供静态方法:提供一个公共的静态方法,用于返回该实例。

二、单例模式的实践

        接下来,我们将通过具体的Java代码示例,演示如何实现单例模式。

2.1 饿汉式

        饿汉式在类加载时就创建实例,因此是线程安全的,但可能会造成资源浪费,如果实例从未被使用过。

// 饿汉式单例模式
public class SingletonEager {// 在类加载时就创建实例private static final SingletonEager INSTANCE = new SingletonEager();// 私有化构造函数private SingletonEager() {// 初始化代码}// 提供全局访问点public static SingletonEager getInstance() {return INSTANCE;}
}


优点

  • 线程安全,因为实例在类加载时就创建。

缺点

  • 即使实例未被使用,也会创建实例,可能造成资源浪费。

2.2 懒汉式(线程不安全)

懒汉式在第一次使用时才创建实例,节省资源,但线程不安全。

// 懒汉式单例模式(线程不安全)
public class SingletonLazyUnsafe {// 静态实例,初始为nullprivate static SingletonLazyUnsafe instance;// 私有化构造函数private SingletonLazyUnsafe() {// 初始化代码}// 提供全局访问点public static SingletonLazyUnsafe getInstance() {if (instance == null) {instance = new SingletonLazyUnsafe();}return instance;}
}


优点

  • 延迟加载,节省资源。

缺点

  • 线程不安全,多个线程同时访问时,可能会创建多个实例。

2.3 懒汉式(线程安全)

通过同步方法或同步代码块,确保线程安全。

同步方法

// 懒汉式单例模式(线程安全,同步方法)
public class SingletonLazySafeSyncMethod {private static SingletonLazySafeSyncMethod instance;private SingletonLazySafeSyncMethod() {// 初始化代码}// 同步方法,确保线程安全public static synchronized SingletonLazySafeSyncMethod getInstance() {if (instance == null) {instance = new SingletonLazySafeSyncMethod();}return instance;}
}


同步代码块

// 懒汉式单例模式(线程安全,同步代码块)
public class SingletonLazySafeSyncBlock {private static SingletonLazySafeSyncBlock instance;private static final Object LOCK = new Object();private SingletonLazySafeSyncBlock() {// 初始化代码}// 同步代码块,提升性能public static SingletonLazySafeSyncBlock getInstance() {if (instance == null) {synchronized (LOCK) {if (instance == null) {instance = new SingletonLazySafeSyncBlock();}}}return instance;}
}


优点

  • 延迟加载,节省资源。
  • 线程安全,通过同步确保只有一个实例。

缺点

  • 同步方法性能较差,因为每次调用都需要同步。
  • 同步代码块虽然性能有所提升,但仍然需要双重检查(Double-Checked Locking),代码相对复杂。

2.4 静态内部类(推荐)

静态内部类方式结合了饿汉式和懒汉式的优点,既实现了延迟加载,又保证了线程安全。

// 静态内部类单例模式(推荐)
public class SingletonStaticInnerClass {// 私有构造函数private SingletonStaticInnerClass() {// 初始化代码}// 静态内部类,负责创建实例private static class Holder {private static final SingletonStaticInnerClass INSTANCE = new SingletonStaticInnerClass();}// 提供全局访问点public static SingletonStaticInnerClass getInstance() {return Holder.INSTANCE;}
}


优点

  • 延迟加载,节省资源。
  • 线程安全,JVM保证静态内部类在第一次使用时才加载。
  • 无需同步,性能较好。

缺点

  • 实现相对复杂,需要理解静态内部类的加载机制。

2.5 枚举(最安全)

枚举单例模式是最简单、最安全的实现方式,天生防止反射和序列化攻击。

// 枚举单例模式(最安全)
public enum SingletonEnum {INSTANCE;// 提供其他方法public void doSomething() {// 实现代码}
}


优点

  • 防止反射攻击,因为JVM禁止通过反射机制创建枚举实例。
  • 防止序列化攻击,因为枚举的序列化机制由JVM保证,不会创建新的实例。
  • 线程安全,由JVM保证。

缺点

  • 实现简单,但可能不适合所有场景,特别是需要继承其他类的场景(Java枚举不能继承非枚举类)。

三、单例模式的注意事项

防止反射攻击:通过私有构造函数和异常处理,防止通过反射机制创建实例。

private Singleton() {if (instance != null) {throw new RuntimeException("Use getInstance() method to get the single instance of this class.");}// 初始化代码
}

防止序列化攻击:通过实现readResolve方法,防止通过序列化机制创建实例。

private Object readResolve() throws ObjectStreamException {return INSTANCE;
}

线程安全:确保在多线程环境下,只有一个实例被创建。

延迟加载:如果实例创建开销较大,考虑使用懒汉式或静态内部类方式。

总结

        单例模式是一种确保某个类只有一个实例,并提供一个全局访问点的设计模式。它通过私有化构造函数、创建静态实例和提供全局访问点来实现。在Java中,有多种实现方式,包括饿汉式、懒汉式(线程不安全、线程安全)、静态内部类和枚举。每种方式都有其优缺点,开发者应根据具体场景选择合适的实现方式。同时,需要注意防止反射和序列化攻击,确保线程安全。通过合理使用单例模式,可以提高代码的可维护性和性能。

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

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

相关文章

zerotier实现内网穿透

zerotier的内网穿透 前言一、zerotier的框架认知二、客户端安装设置1.linux2.windows 前言 摸索了一阵,看了好几篇,没有讲清楚。争取这次说清楚。 一、zerotier的框架认知 先认识一下zerotier的框架,这样如何处理就很好理解了。 首先上zero…

windows11 使用体验记录

好的地方: UI上字体风格貌似更好看了,文件夹增加了多个标签,类似于浏览器既可以打开多个窗口,也可以在同一个窗口中打开多个标签页 不好的地方: 桌面右下角点击日期时间,显示日期,时间呢&…

牛客周赛 Round 70 A-E

本期封面原图 画师村カルキ 牛客周赛 Round 70 五题 A - 小苯晨跑 思路 没啥好说的 代码 // // Created by Swan416 on 2024-12-01 18:57. // #include <bits/stdc.h> #define maxOf(a) *max_element(a.begin(),a.end()) #define minOf(a) *min_element(a.begin(),a.…

C++20: 像Python一样split字符串

概要 Python 的字符串天生支持 split( ) 操作&#xff0c;支持单个字符或字符串作为分隔符。 C 在这方面显得很笨拙&#xff0c;但是在 C20 下经过一番尝试&#xff0c;还是能够提供类似的简洁调用。 Python 代码 s 0,11,336,23,370nums s.split(,) for n in nums:print(n…

oracle数据库日常操作

1、执行SQL语句后不显示PL/SQL procedure successfully completed set feedback off; 2、显示实例名称 echo "set sqlprompt \"_user_connect_identifier> \"" >> $ORACLE_HOME/sqlplus/admin/glogin.sql 3、客户端尝试连接到服务器时发生超时 …

【Redis篇】Hash的认识以及相关命令操作

目录 前言 基本命令 HSET HGET HEXISTS HDEL HKEYS HVALS HGETALL HMGET HLEN HSETNX HINCRBY HINCRBYFLOAT 内部编码 高内聚&#xff0c;低耦合 前言 可以看出&#xff1a; Redis 的 Hash 是一个键&#xff08;key&#xff09;下包含多个字段&#xff08;field…

可解释机器学习 | Python实现LGBM-SHAP可解释机器学习

机器学习 | Python实现GBDT梯度提升树模型设计 目录 机器学习 | Python实现GBDT梯度提升树模型设计基本介绍模型使用参考资料基本介绍 LightGBM(Light Gradient Boosting Machine)是一种基于决策树的梯度提升框架,是一种高效的机器学习模型。SHAP(SHapley Additive exPlan…

<回文判断>

【问题描述】 回文是正读和倒读都一样的句子。读入一个最大长度不超过50个字符的句子&#xff0c;判断其是否是回文。 【输入形式】 输入一个最大长度不超过50个字符的句子 【输出形式】 Yes/No 【输入样例】 abcba 【输出样例】 Yes 【样例说明】 输入abcba&#xff0c;判…

VPP多架构处理器支持

对于转发层面的关键节点&#xff08;node&#xff09;&#xff0c;VPP针对处理器架构编译多份代码&#xff0c;在运行时检测处理器架构&#xff0c;动态确定使用的代码分支。VPP提供两种对多处理器架构的支持&#xff0c;除了节点函数外&#xff0c;还可指定任意函数支持多架构…

算法2--滑动窗口

滑动窗口 滑动窗口经典例题长度最小的子数组无重复字符的最长子串[最大连续1的个数 III](https://leetcode.cn/problems/max-consecutive-ones-iii/description/)[将 x 减到 0 的最小操作数](https://leetcode.cn/problems/minimum-operations-to-reduce-x-to-zero/description…

mysql--二进制安装编译安装yum安装

二进制安装 创建用户和组 [rootlocalhost ~]# groupadd -r -g 306 mysql [rootlocalhost ~]# useradd -r -g 306 -u 306 -d /data/mysql mysql 创建文件夹并添加所属文件用户和组 [rootlocalhost ~]# mkdir -p /data/mysql [rootlocalhost ~]# chown mysql:mysql /data/mysql …

大模型开发和微调工具Llama-Factory-->WebUI

WebUI LLaMA-Factory 支持通过 WebUI 零代码微调大模型。 通过如下指令进入 WebUI llamafactory-cli webui# 如果是国内&#xff0c; # USE_MODELSCOPE_HUB 设为 1&#xff0c;表示模型从 ModelScope 魔搭社区下载。 # 避免从 HuggingFace 下载模型导致网速不畅 USE_MODELSC…

【WPS】【EXCEL】将单元格中字符按照分隔符拆分按行填充到其他单元格

问题&#xff1a;实现如下图的效果 解答&#xff1a; 一、函数 IFERROR(TRIM(MID(SUBSTITUTE($A$2,",",REPT(" ",LEN($A$2))),(ROW(A1)-1)*LEN($A$2)1,LEN($A$2))),"") 二、在单元格C2中填写如下函数 三、全选要填充的单元格并且按CTRLD 函数…

Java有关数组的相关问题

Java中的栈和堆的含义 栈 存储局部变量&#xff1a;栈主要用于存储方法中的局部变量&#xff0c;包括基本数据类型&#xff08;int、double、boolean等&#xff09;和对象的引用&#xff08;不包含对象本身&#xff09;。 遵循后进先出原则&#xff1a;当一个方法被调用时&…

提升阅读体验,Balabolka让文字跃然“声”上

作为一个专业的语音合成工具&#xff0c;Balabolka为用户提供了全方位的文本朗读解决方案。这款软件不仅可以将各类文本实时转换为清晰的语音输出&#xff0c;还能将转换后的音频内容导出为多种主流格式。它强大的兼容性使其能够处理各类电子书和文档格式&#xff0c;让用户可以…

解决`-bash: ./configure:/bin/sh^M:解释器错误: 没有那个文件或目录`的问题

解决`-bash: ./configure:/bin/sh^M:解释器错误: 没有那个文件或目录`的问题 一、错误原因分析二、解决方法方法一:使用`dos2unix`工具方法二:使用`sed`命令方法三:使用`tr`命令方法四:在文本编辑器中转换方法五:在Windows系统中使用适当的工具三、预防措施四、总结在使…

第一部分:基础知识 1. 介绍 --[MySQL轻松入门教程]

什么是MySQL&#xff1f; MySQL 是一种广泛使用的关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;它使用结构化查询语言&#xff08;SQL&#xff09;来管理数据。MySQL 由瑞典公司 MySQL AB 开发&#xff0c;现在归 Oracle Corporation 所有。MySQL 是开源软件…

Flink双流Join

在离线 Hive 中&#xff0c;我们经常会使用 Join 进行多表关联。那么在实时中我们应该如何实现两条流的 Join 呢&#xff1f;Flink DataStream API 为我们提供了3个算子来实现双流 join&#xff0c;分别是&#xff1a; join coGroup intervalJoin 下面我们分别详细看一下这…

C# 匿名类型和匿名方法

在C#中&#xff0c;匿名类型和匿名方法是两个非常有用的特性&#xff0c;它们可以帮助我们更方便地处理数据和简化代码的编写。 一、匿名类型 (Anonymous Types) 匿名类型允许你在不显式定义类的情况下创建对象。编译器会自动为你生成一个类&#xff0c;其属性由你在对象初始…

【真正离线安装】Adobe Flash Player 32.0.0.156 插件离线安装包下载(无需联网安装)

网上很多人声称并提供的flash离线安装包是需要联网才能安装成功的&#xff0c;其实就是在线安装包&#xff0c;而这里提供的是真正的离线安装包&#xff0c;无需联网即可安装成功。 点击下面地址下载离线安装包&#xff1a; Adobe Flash Player 32.0.0.156 for IE Adobe Fla…