【Java多线程】单例模式(饿汉模式和懒汉模式)

目录

单例模式的定义:

饿汉式--单例模式

 定义:

案例: 

优缺点: 

懒汉式--单例模式:

定义:

1)懒汉式单例模式(非线程安全) 

2)线程安全的懒汉式单例模式 (synchronized )

3)双重检查锁定的懒汉式单例模式(线程安全) 


单例模式的定义:

  • 单例模式是一种设计模式,它确保一个类只有一个实例并提供一个全局访问点来访问这个实例。就好像在一个软件系统中,对于某些特定的资源或者对象,只需要一个就足够了,例如数据库连接池、配置文件管理器等。通过单例模式可以更好地控制这些对象的创建和访问,避免创建多个实例导致资源浪费或者数据不一致等问题。

单例模式能保证某个类在程序中只存在唯⼀⼀份实例,⽽不会创建出多个实例. 

要实现单例模式,通常需要做到以下几点

  1. 私有化构造函数,防止外部通过new关键字创建实例。
  2. 提供一个静态的私有变量来保存类的唯一实例。
  3. 提供一个公共的静态方法来获取类的唯一实例,如果实例不存在则创建它。

单例模式具体的实现⽅式有很多.最常⻅的是"饿汉"和"懒汉"两种.


饿汉式--单例模式

定义:

  • 在饿汉式单例模式中,“饿” 体现的是一种急切的状态。就好像一个很饿的人,在看到食物(这里类比于单例对象)的时候,会迫不及待地先把食物拿到手(创建单例对象)。在这个模式下,单例对象在类加载阶段就被创建出来,而不是等到真正需要使用这个对象的时候才去创建。这种方式比较急切,所以被称为 饿汉模式”。

案例: 

class Singleton {
    // 私有静态成员变量,在类加载时就初始化实例
    private static Singleton instance = new Singleton();
    // 私有构造函数,防止外部通过new关键字创建实例
    private Singleton() {}
    // 公共静态方法,用于获取单例实例
    public static Singleton getInstance() {
        return instance;
    }
}
- ** 类加载过程中的创建**
     - 在Java中,类加载是由类加载器(ClassLoader)完成的一个过程。当一个类被首次主动使用(例如创建这个类的实例、访问这个类的静态成员等情况)时,这个类就会被加载。对于上述的`Singleton`类,当`Singleton`类被加载时,`private static Singleton instance = new Singleton();`这行代码就会被执行。 因为类加载机制保证了一个类在一个Java程序中只会被加载一次(在正常情况下),所以`instance`对象也只会被创建一次
   - **访问控制保证单例性**
     - 构造函数`private Singleton()`是私有的。这是非常关键的一点,它防止了外部类通过`new`关键字来创建`Singleton`类的新实例。外部类只能通过`public static Singleton getInstance()`方法来获取单例对象,而这个方法每次返回的都是在类加载阶段就已经创建好的`instance`对象,从而保证了整个系统中 只有一个`Singleton`类的实例存在。

优缺点: 

**优点**
   1 **线程安全**
     - 由于单例对象是在类加载阶段就创建好的,而类加载过程在Java中是线程安全的(由Java虚拟机来保证)。所以在多线程环境下,这种方式可以保证多个线程访问`getInstance`方法时,获取到的都是同一个单例对象,不会出现多个线程创建多个实例的情况。
   2- **实现简单**
     - 从代码量和逻辑复杂度来看,饿汉式单例模式是比较简单的。只需要在类中定义一个私有静态变量并初始化,再提供一个公共静态方法来返回这个变量即可。这种简单的实现方式使得代码易于理解和维护。
**缺点**
   - **可能会造成资源浪费**
     - 如果单例对象的创建过程比较复杂,例如需要进行大量的初始化操作,如加载配置文件、建立网络连接等,并且这个单例对象在程序运行初期可能并不一定需要被使用。那么在类加载阶段就创建这个单例对象可能会导致资源的浪费。就好像提前准备了一顿丰盛的大餐(单例对象),但可能很长时间都没有人来吃(使用单例对象),而准备这顿大餐(创建单例对象)的过程又耗费了很多资源。


懒汉式--单例模式:

定义:

 在懒汉模式下,实例在第一次使用时才进行创建,因此称为“懒汉”,在需要被用的时候被创建,突出一个字“

1)懒汉式单例模式(非线程安全) 

public class LazySingleton {
    // 私有静态变量,用于存储单例对象
    private static LazySingleton instance;
    // 私有构造函数,防止外部通过new关键字创建新的实例
    private LazySingleton() {}
    // 公共的静态方法,用于获取单例对象
    public static LazySingleton getInstance() {
        if (instance == null) {
            // 如果实例还未创建,则创建一个新的实例
            instance = new LazySingleton();
        }
        return instance;
    }
}
  • 这种实现方式在单线程环境下是可以正常工作的。当第一次调用getInstance方法时,会检查instance是否为null。如果是null,就会创建一个LazySingleton类的实例并赋值给instance,然后返回这个实例。之后再调用getInstance方法时,因为instance已经不是null了,所以会直接返回已创建的实例。
  • 存在的问题
    • 在多线程环境下,这种实现方式是不安全的。假设两个线程同时调用getInstance方法,并且此时instancenull。这两个线程都会执行instance = new LazySingleton();这一行代码,从而创建出两个不同的LazySingleton实例,这就违背了单例模式的初衷。

2)线程安全的懒汉式单例模式 (synchronized )

public class ThreadSafeLazySingleton {
    private static ThreadSafeLazySingleton instance;
    private ThreadSafeLazySingleton() {}
    // 使用synchronized关键字修饰方法,保证在多线程环境下的线程安全
    public static synchronized ThreadSafeLazySingleton getInstance() {
        if (instance == null) {
            instance = new ThreadSafeLazySingleton();
        }
        return instance;
    }
}
  • 通过在getInstance方法上添加synchronized关键字,保证了在多线程环境下,同一时刻只有一个线程能够进入这个方法。当一个线程进入getInstance方法并发现instancenull时,它会创建一个新的实例。其他线程如果在这个时候也尝试调用getInstance方法,就会被阻塞,直到第一个线程完成实例的创建并返回。
  • 存在的问题
    • 这种方式虽然保证了线程安全,但是性能较差。因为每次调用getInstance方法都需要获取锁,即使实例已经创建完成,这种不必要的同步操作会在高并发场景下成为性能瓶颈。

3)双重检查锁定的懒汉式单例模式(线程安全) 

public class DoubleCheckedLockingSingleton {
    // 使用volatile关键字保证变量的可见性和禁止指令重排序
    private static volatile DoubleCheckedLockingSingleton instance;
    private DoubleCheckedLockingSingleton() {}
    public static DoubleCheckedLockingSingleton getInstance() {
        if (instance == null) {
            // 第一次检查,提高性能,避免不必要的同步操作
            synchronized (DoubleCheckedLockingSingleton.class) {
                if (instance == null) {
                    // 第二次检查,确保在同步块内也不会创建多个实例
                    instance = new DoubleCheckedLockingSingleton();
                }
            }
        }
        return instance;
    }
}
  • 首先,if (instance == null)这一检查在同步块外进行,这是第一次检查。如果instance已经不是null就可以直接返回实例,避免了进入同步块,从而提高了性能
  • 当第一次检查instancenull时,线程会进入同步块。在同步块内,又进行了一次if (instance == null)检查,这是第二次检查。这是为了防止在多个线程同时通过第一次检查后,只有一个线程能够进入同步块创建实例,其他线程在等待这个线程完成创建后,直接获取已创建的实例,而不会再次创建
  • 使用volatile关键字是非常关键的。在 Java 中,指令重排序可能会导致instance变量在没有完全初始化的情况下就被其他线程看到。volatile关键字可以保证变量的可见性,并且禁止指令重排序,确保了单例模式的正确性。

结语: 写博客不仅仅是为了分享学习经历,同时这也有利于我巩固知识点,总结该知识点,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进。同时也希望读者们不吝啬你们的点赞+收藏+关注,你们的鼓励是我创作的最大动力! 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【C++】类与对象的基础概念

目录: 一、inline 二、类与对象基础 (一)类的定义 (二)访问限定符 (三)类域 (四)实例化概念 正文 一、inline 在C语言的学习过程中,大家肯定了解过宏这个概…

matlab实现主成分分析方法图像压缩和传输重建

原创 风一样的航哥 航哥小站 2024年11月12日 15:23 江苏 为了研究图像的渐进式传输技术,前文提到过小波变换,但是发现小波变换非常适合传输缩略图,实现渐进式传输每次传输的数据量不一样,这是因为每次变换之后低频成分大约是上一…

python成长技能之网络编程

文章目录 一、初识Socket1.1 什么是 Socket?1.2 socket的基本操作1.3 socket常用函数 二、基于UDP实现客户端与服务端通信三、基于TCP实现客户端与服务端通信四、使用requests模块发送http请求 一、初识Socket 1.1 什么是 Socket? Socket又称"套接字",…

ROM修改进阶教程------安卓14 安卓15去除app签名验证的几种操作步骤 详细图文解析

在安卓14 安卓15的固件中。如果修改了系统级别的app。那么就会触发安卓14 15的应用签名验证。要么会导致修改的固件会进不去系统,或者进入系统有bug。博文将从几方面来解析去除安卓14 15应用签名验证的几种方法。 💝💝💝通过博文了解: 1💝💝💝-----安卓14去除…

[Docker#6] 镜像 | 常用命令 | 迁移镜像 | 压缩与共享

目录 Docker 镜像是什么 生活案例 为什么需要镜像 镜像命令详解 实验 1.一些操作 1. 遍历查看镜像 2. 查看镜像仓库在本地的存储信息 进入镜像存储目录 查看 repositories.json 文件 3. 镜像过滤 4. 下载镜像时的分层 实战一:离线迁移镜像 实战二&…

「QT」几何数据类 之 QVector3d 三维向量类

✨博客主页何曾参静谧的博客📌文章专栏「QT」QT5程序设计📚全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasolid…

人工智能(AI)对于电商行业的变革和意义

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/402a907e12694df5a34f8f266385f3d2.png#pic_center> 🎓作者简介:全栈领域优质创作者 🌐个人主页:百锦再新空间代码工作室 📞工作室:新空间代…

物联网设备研究——分配推理负载的联合学习方法

概述 物联网(IoT)的最新发展导致人工智能模型被嵌入到传感器和智能手机等终端设备中。这些模型是根据每个设备的存储容量和计算能力定制的,但重点是在终端侧进行本地推理,以降低通信成本和延迟。 然而,与部署在边缘服…

CentOS Stream 9设置静态IP

CentOS Stream 9设置静态IP CentOS Stream 9作为CentOS Stream发行版的下一个主要版本,已经发布有一段时间,但与目前广泛使用的CentOS7有较大区别。安装试用Stream 9的过程中,就发现设置静态IP的方式和CentOS7/8差别较大,在此记录…

【嵌入式】ESP32开发(一)ESP-IDF概述

文章目录 1 前言2 IDF环境配置3 在VS Code中使用IDF3.1 使用ESP-IDF例程3.2 底部按钮的作用【重要!】3.3 高级用法4 ESP-IDF框架分析5 从零开始创建一个项目5.1 组件(component)6 主要参考资料7 遇到的一些问题与解决办法8 对于ESP-IDF开发的一些感受1 前言 对于ESP32的开发…

基于Multisim水箱水位控制系统仿真电路(含仿真和报告)

【全套资料.zip】水箱水位控制系统仿真电路Multisim仿真设计数字电子技术 文章目录 功能一、Multisim仿真源文件二、原理文档报告资料下载【Multisim仿真报告讲解视频.zip】 功能 1.在水箱内的不同高度安装3根金属棒,以感知水位变化情况, 液位分1&…

解读Nature:Larger and more instructable language models become less reliable

目录 Larger and more instructable language models become less reliable 核心描述 核心原理 创新点 举例说明 大模型训练,微调建议 Larger and more instructable language models become less reliable 这篇论文的核心在于对大型语言模型(LLMs)的可靠性进行了深入…

zabbix监控端界面时间与服务器时间不对应

1. 修改系统时间 # tzselect Please select a continent, ocean, "coord", or "TZ".1) Africa2) Americas3) Antarctica4) Asia5) Atlantic Ocean6) Australia7) Europe8) Indian Ocean9) Pacific Ocean 10) coord - I want to use geographical coordina…

ubuntu20.04安装FLIR灰点相机BFS-PGE-16S2C-CS的ROS驱动

一、Spinnaker 安装 1.1Spinnaker 下载 下载地址为: https://www.teledynevisionsolutions.com/support/support-center/software-firmware-downloads/iis/spinnaker-sdk-download/spinnaker-sdk–download-files/?pnSpinnakerSDK&vnSpinnakerSDK 在上述地址中…

Windows配置JDK

1、解压 下载以后解压,放在一个没有中文路径和没有空格的目录,如下图: 2、配置Java环境 1)、点击左下角windows图标,输入huanjing(或者path),打开环境变量配置 如图: …

Unity教程(十八)战斗系统 攻击逻辑

Unity开发2D类银河恶魔城游戏学习笔记 Unity教程(零)Unity和VS的使用相关内容 Unity教程(一)开始学习状态机 Unity教程(二)角色移动的实现 Unity教程(三)角色跳跃的实现 Unity教程&…

HCIP-HarmonyOS Application Developer 习题(二十三)

1、(多选)端云一体化已经集成以下哪些服务SDK。 A、云函数 B、云数据库 C、云存储 D、云托管 答案:AB 分析:云开发即为应用开发云侧工程,目前包含云函数与云数据库工程。 2、(多选)Entry下的m…

图数据库 | 5、图数据库三大组件之一 之 图计算 (下)

书接上文:图数据库 | 4、图数据库三大组件之一 ——图计算 (上)-CSDN博客 结合计算效率来评估与设计图计算所需的数据结构。 存储低效性或许是相邻矩阵或关联矩阵等数据结构的最大缺点,尽管它有着O(1)的访问时间复杂度。例如通过…

Android OpenGL ES详解——纹理:纹理过滤GL_NEAREST和GL_LINEAR的区别

目录 一、概念 1、纹理过滤 2、邻近过滤 3、线性过滤 二、邻近过滤和线性过滤的区别 三、源码下载 一、概念 1、纹理过滤 当纹理被应用到三维物体上时,随着物体表面的形状和相机视角的变化,会导致纹理在渲染过程中出现一些问题,如锯齿…

【java】java通过s3访问ceph报错

1.报错信息、背景 工作中起了几个访问ceph的服务pod节点,一段时间后1个节点一直报错Unable to execute HTTP request: Timeout waiting for connection from pool,详细i信息如下图片,有且仅有1个节点报错,其他节点访问正常。看日志…