<JavaEE> 经典设计模式之 -- 单例模式(“饿汉模式”和“懒汉模式”实现单例模式)

目录

一、单例模式概述

二、“饿汉模式”实现单例模式

三、“懒汉模式”实现单例模式

3.1 单线程下的“懒汉模式”

3.2 多线程下的“懒汉模式”


一、单例模式概述

1)什么是单例模式?

单例模式是一种设计模式。

单例模式可以保证某个类在程序中只存在唯一实例,即不允许创建多份实例。

使用单例模式,上述要求就得到了检查和校验。

2)单例模式的实现形式

单例模式可以通过很多种方法实现,“饿汉模式”和“懒汉模式”是其中最基础的两种,本文只介绍这两种实现。

二、“饿汉模式”实现单例模式

通过代码演示“饿汉模式”实现的单例模式:

class Singleton{//新建一个唯一实例;private static Singleton instance = new Singleton();//方法返回唯一实例;public static Singleton getInstance() {return instance;}//将构造方法私有化;private Singleton() { }
}
1)上述代码做了什么?

创建了一个被 static 修饰的实例,这个实例成为了类属性。类对象只会有一个,这个类属性也只会有一个。

私有化构造方法,外部无法 new 新的实例,只能通过 get 方法获取唯一的那一个 instance。
2)为什么叫做“饿汉模式”?

上述代码中,实例是类属性。类属性在类加载的时候就创建了,创建时机早,十分“迫切”,因此称为“饿汉模式”。

代码证明“饿汉模式”返回的实例是唯一的:

public class Singleton_Demo0 {public static void main(String[] args) {//想直接new对象,就会报错;//Singleton instance = new Singleton();//两次调用getInstance()方法并分别赋值;Singleton instance1 = Singleton.getInstance();Singleton instance2 = Singleton.getInstance();//对比两个变量,发现是同一实例;if(instance1 == instance2){System.out.println("两个对象是同一个对象");}}
}//运行结果:
两个对象是同一个实例
3)“饿汉模式”的单例模式在多线程下是线程安全的吗?

上述代码中,get 方法返回的是已经创建好的实例,这个操作本质上只是一个“读操作”,多个线程读取同一个变量并不会造成线程不安全。

因此“饿汉模式”的单例模式在多线程下是线程安全的。

三、“懒汉模式”实现单例模式

3.1 单线程下的“懒汉模式”

通过代码演示懒汉模式现的单例模式:

class Singleton{//声明一个变量作为类属性;private static Singleton instance = null;//判断变量是否为null,是则创建实例后返回,否则返回;public static Singleton getInstance() {if(instance == null){instance = new Singleton();}return instance;}//将构造方法私有化;private Singleton() { }
}
1)上述代码做了什么?

声明了一个类属性。类对象只会有一个,这个类属性也只会有一个。

私有化构造方法,外部无法 new 新的实例,只能通过 get 方法获取唯一的那一个 instance。
get 方法中根据变量是否为 null 判断是否应该创建实例。
2)为什么叫做“懒汉模式”?

上述代码中,实例是在程序员第一次调用 get 方法后才创建的,创建时机较晚,或者根本不用创建,因此称为“懒汉模式”。

3.2 多线程下的“懒汉模式”

1)单线程下的“懒汉模式”在多线程下是线程安全的吗?

答案是否定的,单线程下的“懒汉模式”在多线程下是线程不安全的,我们可以从以下两个方面分析:

“原子性”:

上述代码中判断变量是否为空的代码 —— if(instance == null),和实例化代码 —— 

instance = new Singleton(),并非是“原子”的。在多线程环境下,这就可能导致线程不安全。

可以使用 synchronized 关键字,将这两句代码加锁,解决这个问题。

内存可见性和指令重排序:

因为 instance 是一个被 static 修饰的共享数据,而且编译器内部可能对实例化的代码 —— new Singleton(),进行了编译器优化。

这就无法保证内存的可见性和指令的顺序执行,因此在多线程环境下可能导致线程不安全。

可以使用 volatile 关键字,对共享数据 instance 进行修饰,解决这个问题。

使用以上两个关键字的原因和方式,详细请参考以下博客:

阅读指针 -> 《synchronized 关键字 和 volatile 关键字》<JavaEE> synchronized关键字和锁机制 -- 锁的特点、锁的使用、锁竞争和死锁、死锁的解决方法-CSDN博客文章浏览阅读70次。介绍了 synchronized 关键字 和 锁机制,其中重点介绍了锁的特点、使用方法和死锁的相关内容。https://blog.csdn.net/zzy734437202/article/details/134742168<JavaEE> volatile关键字 -- 保证内存可见性、禁止指令重排序-CSDN博客文章浏览阅读59次。简单介绍什么是内存可见性和指令重排序。volatile关键字可以将这两种编译器优化强制关闭。https://blog.csdn.net/zzy734437202/article/details/134757070

2)“懒汉模式”在多线程下应该怎么编写?

根据上述分析,根据单线程模式下的“懒汉模式”进行改进。

方法如下:

增加 volatile 关键字对共享数据进行修饰。

为判断是否为 null 和 实例化的代码加锁,使这两句代码称为“原子”。

增加 volatile 关键字对共享数据进行修饰:

private volatile static Singleton instance = null;

为判断是否为 null 和 实例化的代码加锁,使这两句代码称为“原子”:

    public static Singleton getInstance() {synchronized (locker){if(instance == null){instance = new Singleton();}}return instance;}
3)“双重校验锁”

我们再仔细分析一下上述的 get 方法。

假设程序需要多次调用这个 get 方法,那么每一次进入都会进行加锁,加锁是会增加系统开销的。

那么是否真的有必要每次都加锁呢?

当 get 方法被第一次调用,实例就会被创建,那么后续再调用这个 get 方法时,返回实例就好了,加锁部分的代码块,完全可以不用执行。

在加锁的代码块之外,再增加一个if(instance == null)进行判断,那么实例在被创建之后,也就不会再进入加锁的代码块中了。

我们成功利用“双重校验锁”,优化了程序。

代码演示“双重校验锁”优化后的 get 方法:

    public static Singleton getInstance() {//这个if用于判断是否需要加锁;if(instance == null){synchronized (locker){//这个if用于判断是否需要新建实例;if(instance == null){instance = new Singleton();}}}return instance;}

经过以上的完善和优化,我们终于可以写出在多线程下保证线程安全的“懒汉模式”单例模式了:

class Singleton{//声明一个变量作为类属性;private volatile static Singleton instance = null;private static final Object locker = new Object();//判断变量是否为null,是则创建实例后返回,否则返回;public static Singleton getInstance() {//这个if用于判断是否需要加锁;if(instance == null){synchronized (locker){//这个if用于判断是否需要新建实例;if(instance == null){instance = new Singleton();}}}return instance;}//将构造方法私有化;private Singleton() { }
}

阅读指针 -> 《经典设计模式之 -- 使用阻塞队列实现“生产者-消费者模型”》

<JavaEE> 经典设计模式之 -- 使用阻塞队列实现“生产者-消费者模型”-CSDN博客自己实现了的阻塞队列,介绍了经典的设计模式“生产者-消费者模型”。https://blog.csdn.net/zzy734437202/article/details/134807241

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

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

相关文章

【Java数据结构 -- 顺序表】

List和ArrayList与顺序表 一. List1.1 List介绍2.1 常见接口介绍3.1 List的使用 二. ArrayList与顺序表1.线性表2.顺序表2.1 接口的实现2.2 顺序表的创建2.3 顺序表的打印2.4 顺序表的插入2.5 顺序表的按索引位置插入数据2.6 判断顺序表是否包含某个数2.7 返回顺序表某个数的索…

2023-12学习笔记

1.NonNull要手动写无参构造器 这是一个我今天研究了很久的问题,开始不知道原因是在这里,还在那想是不是Data覆盖了无参构造,结果当然不是。先说下解决历程 1.问题起因 通过RequestBody接收前端报文的时候报错,大致是说我构造方…

python中tkinter实现GUI程序:三个实例

python中tkinter实现GUI程序 写在最前面Python中使用Tkinter实现GUI程序的基本元素Tkinter简介基本元素1. 根窗口(Root Window)2. 小部件(Widgets)3. 布局管理4. 事件处理 1.用 tkinter实现一个简单的 GUI程序,单击“click”按钮&…

Java项目-瑞吉外卖Day3

填充公共字段: 目的:由于某些属性,例如createdTime这些需要填充的字段会在多个地方出现,所以考虑使用公共字段自动填充的办法减少重复代码。 在对应属性上加入TableField注解。通过fill字段表明策略,是插入/更新的时候…

如何搭建废品上门回收小程序

如今,随着环境保护意识的增强,废品的回收和再利用变得越来越重要。为了方便人们进行废品回收,搭建一个废品上门回收的小程序成为了一个不错的选择。本文将介绍如何从零开始搭建一个废品上门回收小程序。 …

【Go自学版】02-goroutine

利用时间片分割进程,致使宏观上A,B,C同时执行(并发) CPU利用率包含了执行和切换,进程/线程的数量越多,切换成本也会增大 最大并行数:GOMAXPROCS work stealing: 偷其他队列的G hand off: 当前G1阻塞&#…

css 修改滚动条样式,解决Windows浏览器中滚动条不美观问题

Windows环境中的浏览器中滚动条默认是直接显示了,不管光标是否进入该区域,这样就很不美观,如下图: 之前样式为 .well {display: block;background-color: #f2f2f2;border: 1px solid #ccc;margin: 5px;width: calc(100% - 12px);h…

mycat部署和配置读写分离(二)

说明: MyCAT 是使用 JAVA 语言进行编写开发,使用前需要先安装 JAVA 运行环境(JRE),由于 MyCAT 中使用了 JDK7 中的一些特性,所以要求必须在 JDK7 以上的版本上运行。 1. jdk1.8安装 详见jdk环境安装 2. Mysql安装 详见mysql8.0.11源码安装…

【lesson11】数据类型之string类型

文章目录 数据类型分类string类型set类型测试 enum类型测试 string类型的内容查找找所有女生(enum中)找爱好有游泳的人(set中)找到爱好中有足球和篮球的人 数据类型分类 string类型 set类型 说明: set:集…

SL9008 3.6-60V输入 LED降压恒流芯片 内置MOS管 带PWM调光

SL9008是一款内置MOS管、具有PWM调光功能的LED降压恒流芯片,适用于3.6-60V的输入电压范围。它采用了先进的电路设计,确保了高效率和长寿命,同时具有宽电压输入范围和优异的负载调整率。 SL9008的主要特点包括: 1. 宽输入电压范围&…

HarmonyOS4.0开发应用——【ArkUI组件使用】

ArkUI组件使用 这里会详细演示以下组件使用: ImageTextTextInputButtonSliderColumn&&RowList自定义组件以及相关函数使用 Image 可以是网络图片、可以是本地图片、也可以是像素图 Image("https://ts1.cn.mm.bing.net/th?idOIP-C.cYA-_PINA-ND9OeBaolDTwHaHa&…

CLIP在Github上的使用教程

CLIP的github链接:https://github.com/openai/CLIP CLIP Blog,Paper,Model Card,Colab CLIP(对比语言-图像预训练)是一个在各种(图像、文本)对上进行训练的神经网络。可以用自然语…

鸿蒙HarmonyOS(ArkTS)语法 声明变量及注意事项

好 今天我们来看一个基础的harmonyOS语法 变量声明 这里 我们还是用 ArkTS项目 我们声明变量的语法并不是ArkTS的 而是 javaScript 和 TypeScript的 可以看一下下面一张图 js是最初弱类型语言 于是TS作为js的副类 是一种更严谨的数据限定语法 而ArkTS 是TS的改良版 其实我们…

一篇文章教你快速弄懂 web自动化测试中的三种等待方式

前言 现在的网页很多都是动态加载的,如果页面的内容发生了改变,就需要时间来渲染。在咱们做web自动化测试的时候,由于代码是自动执行的,代码在执行的时候,有可能上一步操作而加载的元素还没加载出来,就会报…

配置本地端口镜像示例(1:1)

本地端口镜像简介 本地端口镜像是指观察端口与监控设备直接相连,观察端口直接将镜像端口复制来的报文转发到与其相连的监控设备进行故障定位和业务监测。 配置注意事项 观察端口专门用于镜像报文的转发,因此不要在上面配置其他业务,防止镜像…

建筑学VR虚拟仿真情景实训教学

首先,建筑学VR虚拟仿真情景实训教学为建筑学专业的学生提供了一个身临其境的学习环境。通过使用VR仿真技术,学生可以在虚拟环境中观察和理解建筑结构、材料、设计以及施工等方面的知识。这种教学方法不仅能帮助学生更直观地理解复杂的建筑理论&#xff0…

记录 | ubuntu源码编译安装/更新boost版本

一、卸载当前的版本 1、查看当前安装的boost版本 dpkg -S /usr/include/boost/version.hpp通过上面的命令,你就可以发现boost的版本了,查看结果可能如下: libboost1.54-dev: /usr/include/boost/version.hpp 2、删除当前安装的boost sudo …

记录 | 使用samba将ubuntu文件夹映射到windows实现共享文件夹

一、ubuntu配置 1. 安装 samba samba 是在 Linux 和 UNIX 系统上实现 SMB 协议的一个免费软件,由服务器及客户端程序构成。SMB(Server Messages Block,信息服务块)是一种在局域网上共享文件和打印机的一种通信协议。 sudo apt-…

Excel COUNT类函数使用

目录 一. COUNT二. COUNTA三. COUNTBLANK四. COUNTIF五. COUNTIFS 一. COUNT ⏹用于计算指定范围内包含数字的单元格数量。 基本语法 COUNT(value1, [value2], ...)✅统计A2到A7所有数字单元格的数量 ✅统计A2到A7,B2到B7的所有数字单元格的数量 二. COUNTA ⏹计…

大数据分析与应用实验任务十一

大数据分析与应用实验任务十一 实验目的 通过实验掌握spark Streaming相关对象的创建方法; 熟悉spark Streaming对文件流、套接字流和RDD队列流的数据接收处理方法; 熟悉spark Streaming的转换操作,包括无状态和有状态转换。 熟悉spark S…