【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关键字可以保证变量的可见性,并且禁止指令重排序,确保了单例模式的正确性。

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

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

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

相关文章

计算机网络——HTTP篇

基础篇 IOS七层网络模型 TCP/IP四层模型? 应⽤层:位于传输层之上,主要提供两个终端设备上的应⽤程序之间的通信,它定义了信息交换的格式,消息会交给下⼀层传输层来传输。 传输层的主要任务就是负责向两台设备进程之间…

Unity插件NodeCanvas之行为树的详细教程

文章目录 前言叶节点 Leafs1、行为 Action2、判断 Condition控制组件 Composites1、顺序执行器 Sequencer2、选择执行器 Selector3、概率选择执行器 Probability Selector4、权重选择执行器 Priority Selector5、平行执行器 Parallel6、轮流选择器 Flip Selector7、完整执行器 …

Qt 使用QTreeView显示并动态的增删改查JSON文件数据

文章目录 效果图概述部分代码总结 效果图 概述 本案例在此开源项目QJsonModel的基础上实现,动态的生成并操作JSON数据,QJsonModel是一个基于QAbstractItemModel的JSON数据模型,它提供了一种简单的方式来将JSON数据可视化,功能简单…

计算机专业可以报考公务员吗?都有哪些职位?

C哥专业提供——计软考研院校选择分析专业课备考指南规划 计算机专业是历年来考公的热门专业,岗位较多,且研究生进入体制内直接是副科级待遇,铁饭碗还是很香的! 25国考计算机专业可以选择的岗位 招聘人数:学硕>专硕…

视频智能分析平台LiteAIServer算法定制未戴安全帽检测技术:智能安防领域的新篇章

在当今这个科技日新月异的时代,人工智能(AI)和计算机视觉技术的飞速发展正在以前所未有的速度改变我们的生活、工作乃至整个社会的面貌。在建筑行业中,安全问题一直是重中之重,而工人未戴安全帽的问题更是工地安全管理…

C++ 优先算法 —— 四数之和(双指针)

目录 题目:四数之和 1. 题目解析 2. 算法原理 Ⅰ. 暴力枚举 Ⅱ. 双指针算法 不漏的处理: 去重处理: 3. 代码实现 Ⅰ. 暴力枚举 Ⅱ. 双指针算法 题目:四数之和 1. 题目解析 题目截图: 这道题与三数之和&am…

思考:linux Vi Vim 编辑器的简明原理,与快速用法之《 7 字真言 》@ “鱼爱返 说 温泉啊“ (**)

Linux vi/vim | 菜鸟教程 https://zhuanlan.zhihu.com/p/602675406 Linux Vim编辑器的基本使用_vim文本编辑器-CSDN博客 这里提出使用 vi / vim 进行简单的编辑操作的原因,主要是在容器镜像中,普遍都是使用这个。 在 linux 服务器应用场景&#x…

C# 通俗易懂的介绍基础知识(七)——栈Stack(从日常生活开始讲解)

目录 一、前言 二、栈是排列方式 三、栈的单词 四、程序中的栈 五、栈的方法 1.声明并初始化栈 2.往栈里放东西(学名:入栈) 3.从栈往外拿东西 (学名:出栈) 4.清空栈 5.遍历 Stack 6.获取Stack的长…

React Query在现代前端开发中的应用

💓 博客主页:瑕疵的CSDN主页 📝 Gitee主页:瑕疵的gitee主页 ⏩ 文章专栏:《热点资讯》 React Query在现代前端开发中的应用 React Query在现代前端开发中的应用 React Query在现代前端开发中的应用 引言 React Query …

面试题之---解释一下原型和原型链

实例化对象 和普调函数一样,只不过调用的时候要和new连用(实例化),不然就是一个普通函数调用 function Person () {} const o1 new Person() //能得到一个空对象 const o2 Person() //什么也得不到,这就是普通的…

Java项目实战II基于微信小程序的助农扶贫的设计与实现(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发,CSDN平台Java领域新星创作者,专注于大学生项目实战开发、讲解和毕业答疑辅导。 一、前言 随着移动互联网技术的快速发展&#xf…

【Linux】阿里云服务器搭建gradio实例

文章目录 0 前言1 整体结构2 Python开放环境搭建2.1 SSL安装2.2 python安装 3 开发3.1 工具使用3.2 gradio3.3 langchain 4 nginx部署4.1 下载安装4.2 配置访问gradio 5 服务编写6 扩展:安装软件步骤总结 0 前言 因为课程原因,需要做一个大模型相关的大作…

vue3中如何实现标准元素 拖动 功能 【收藏备用】

最近在用vue3做一个企业后台管理系统的项目,在登录页面的时候需要用户滑动滑块来获取验证码登录系统 用到了元素拖放 这里也顺便记录一下 如何使用的. 目录 1.功能介绍 2.代码部分 3 实现过程 3.1 设置可拖动元素 3.2 拖动什么 3.3 放到何处 3.4 进行放置 1.功能介绍…

小水电远程集控运维系统简介及应用价值

一、政策背景 2023年7月14日水利部办公厅印发了《智能化小型水电站技术指南( 试行)》和(《小水电集控中心技术指南( 试行)》两个指导性文件,明确要求实施小水电绿色改造和现代化提升工程,推进建设智能集约的现代化小水电。 二、系统概述 小…

公司电脑加全屏水印怎么加(怎么打水印满屏)?4个方法精选!包教包会!

在企业管理中,为了保护公司机密信息的安全,给公司电脑添加全屏水印已成为一种常见的安全措施。 全屏水印不仅可以震慑潜在的窥探者,还能在信息不慎泄露时提供追溯线索。 那么,如何给公司电脑添加全屏水印呢? 以下是4…

AI大模型开发架构设计(18)——基于大模型构建企业知识库案例实战

文章目录 1 LLM 大模型在工作中的实际应用以及局限性LLM 大模型工作中实际应用大模型2点局限性 2 基于大模型和向量数据库的企业级知识库架构剖析向量数据库向量数据库选型知识库文档检索增强(Retrieval Augmented Generation)向量数据库应用技术总体架构向量数据库应用离线索引…

10款PDF合并工具的使用体验与推荐!!!

在如今的信息洪流中,我们几乎每个人都被淹没在大量的数字文档之中。无论是学生、教师还是职场人士,我们都需要高效地管理和处理这些文档。而PDF文件,凭借其跨平台的稳定性和通用性,成了最常用的文档格式之一。我们经常需要处理、编…

Redis做分布式锁

(一)为什么要有分布式锁以及本质 在一个分布式的系统中,会涉及到多个客户端访问同一个公共资源的问题,这时候我们就需要通过锁来做互斥控制,来避免类似于线程安全的问题 因为我们学过的sychronized只能对线程加锁&…

IntelliJ+SpringBoot项目实战(四)--快速上手数据库开发

对于新手学习SpringBoot开发,可能最急迫的事情就是尽快掌握数据库的开发。目前数据库开发主要流行使用Mybatis和Mybatis Plus,不过这2个框架对于新手而言需要一定的时间掌握,如果快速上手数据库开发,可以先按照本文介绍的方式使用JdbcTemplat…

12、Linux系统的网络基本设置

查看网络接口信息ifconfig ip addr/ip a #简单查看网络接口信息 ifconfig #表示只显示当前活跃的设备接口信息 ifconfig -a #查看当前主机所有的(all)网络设备,包括未运行的设备。 如我们查看本机网卡ens33的…