设计模式——单例模式详解
目录
- 设计模式类型
- 单例模式
- 单例模式方式
- 饿汉式
- 静态常量方式
- 静态代码块形式
- 懒汉式
- 线程不安全(不推荐)
- 懒汉式优化(不推荐)
- 双重检查(推荐方式)
- 静态内部类(推荐方式)
- 枚举方式(推荐方式)
- 单例模式在JDK中的使用
- 单例模式注意事项和细节说明
- 单例模式的使用场景
设计模式类型
设计模式分为三种类型,共23种
- 创建型模式: 单例模式,抽象工厂模式,原型模式,建造者模式,工厂模式
- 结构性模式: 适配器模式,桥接模式,装饰模式,组合模式,外观模式,享元模式,代理模式
- 行为型模式: 模板方法模式,命令模式,访问者模式,迭代器模式,观察者模式,中介者模式,备忘录模式,解释器模式,状态模式,策略模式,责任链模式
单例模式
所谓类的单例设计模式,就是采取一定的方法保证在整个系统中,对某个类 只能存在一个对象实例,并且该类值提供一个取得对象实例的方法(静态方法)
单例模式方式
饿汉式
静态常量方式
package 单例模式.饿汉式;/*** @author Han* @data 2023/10/27* @apiNode*/
public class Test1 {public static void main(String[] args) {Obj obj1 = Obj.getObj();Obj obj2 = Obj.getObj();// 因为是单例模式所以这两个对象是同一个,所以返回trueSystem.out.println(obj1 == obj2);}
}class Obj {// 创建一个私有的静态对象private final static Obj obj = new Obj();// 将构造方法私有化private Obj() {}// 提供静态公共方法将这个对象返回public static Obj getObj() {return obj;}}
静态代码块形式
package 单例模式.饿汉式;/*** 静态代码块方式** @author Han* @data 2023/10/27* @apiNode*/
public class Test2 {public static void main(String[] args) {Obj2 obj21 = Obj2.getObj();Obj2 obj22 = Obj2.getObj();// 因为是单例模式所以这两个对象是同一个,所以返回trueSystem.out.println(obj21 == obj22);}
}class Obj2 {// 声明一个私有的静态对象private static Obj2 obj2;static {// 在静态代码块中创建对象obj2 = new Obj2();}// 将构造方法私有化private Obj2() {}// 提供静态公共方法将这个对象返回public static Obj2 getObj() {return obj2;}}
优缺点说明:
- 优点:写法简单,在类装载是完成实例化,避免了线程同步问题
- 却爱:在类转载的时候就完成实例化,没有达到lazy loading的效果,如果从始至终都没有用过这个实例,则会造成内存的浪费
- 这种法方式居于classloder机制避免了多线程的同步问题,不过 obj是在类装载是就实例化了,在单例模式中大多都是调用getObj方法
- 结论:这种单例模式可用,可能会造成内存浪费
懒汉式
线程不安全(不推荐)
package 单例模式.懒汉式;/*** 这种方式是线程不安全的* 原因在于在多线程状态下,if判断条件,* 可以能会出现第一个对象还未创建,第二个线程就去判断* 而发生创建多个对象的情况* @author Han* @data 2023/10/27* @apiNode*/
public class Test1 {public static void main(String[] args) {Obj obj1 = Obj.getObj();Obj obj2 = Obj.getObj();System.out.println(obj1 == obj2);}
}
class Obj {// 声明一个静态对象private static Obj obj;// 私有化构造函数private Obj() {}// 提供获取单例对象的方法public static Obj getObj(){// 如果还没有创建对象再去创建,不会发生内存的浪费if (obj == null) {obj = new Obj();}return obj;}
}
优缺点说明
- 起到了lazy loading的效果,但是只能在单线程下使用
- 如果在多线程下使用,一个线程进入了
if(obi == null)
判断语句块,还未来的及王往下执行,另一个线程也通过这个判断语句,这是会发生创建多个实例的错误,所以在多线程环境下不能使用 - 结论:在实际开发中,不要使用这种方式
懒汉式优化(不推荐)
优化,加同步方法,解决线程不安全问题 存在效率问题,
package 单例模式.懒汉式;/*** 这种方式虽然解决了线程安全问题* 但是效率很低* @author Han* @data 2023/10/27* @apiNode*/
public class Test2 {public static void main(String[] args) {Obj2 obj1 = Obj2.getObj();Obj2 obj2 = Obj2.getObj();System.out.println(obj1 == obj2);}
}class Obj2 {// 声明一个静态对象private static Obj2 obj;// 私有化构造函数private Obj2() {}// 提供获取单例对象的方法// 加入了同步处理的代码,解决线程安全问题public static synchronized Obj2 getObj(){// 如果还没有创建对象再去创建,不会发生内存的浪费if (obj == null) {obj = new Obj2();}return obj;}
}
优缺点说明
- 解决了线程安全问题
- 效率太低,每个线程在获得类的实例的时候,执行getObj方法都要进行同步,但是这个方法只需要执行一次实例化代码就够了,后面想要获取该实例直接return就行了,方法进行同步效率太低
- 结论:在实际开发中,不推荐使用这中方式
双重检查(推荐方式)
package 单例模式.双重检查;import com.sun.org.apache.xpath.internal.operations.Variable;/*** 双重检查* 解决线程安全问题,并且支持懒加载** @author Han* @data 2023/10/27* @apiNode*/
public class Test1 {public static void main(String[] args) {Obj obj1 = Obj.getObj();Obj obj2 = Obj.getObj();System.out.println(obj1 == obj2);}
}class Obj {// 声明一个静态对象// 并且使Obj的对象的改变立即更新到内存,在下面的双重检查中判断是否为nulprivate static volatile Obj obj;// 私有化构造函数private Obj() {}// 提供获取单例对象的方法public static Obj getObj() {// 如果还没有创建对象再去创建,不会发生内存的浪费if (obj == null) {// 同步代码块synchronized (Obj.class) {// 再一次检查是否为nullif (obj == null) {obj = new Obj();}}}return obj;}
}
优缺点说明
- 双重检查概念是多线程开发中常用到的,如代码中所示,我们进行了两次
if (obj == null )
的检查,这样就可以保证线程安全 - 这样,实例化代码也只执行一次,后面再次访问时,判断if 直接return实例化对象,也避免了反复进行方法同步
- 线程安全,延迟加载,效率较高
- 结论:在开发中,推荐使用这种单例设计模式
静态内部类(推荐方式)
package 单例模式.静态内部类;import com.sun.org.apache.xpath.internal.operations.Variable;/*** 静态内部类** @author Han* @data 2023/10/27* @apiNode*/
public class Test1 {public static void main(String[] args) {Obj obj1 = Obj.getObj();Obj obj2 = Obj.getObj();System.out.println(obj1 == obj2);}
}class Obj {// 私有化构造函数private Obj() {}// 使用静态内部类public static Obj getObj() {// 使用静态内部类中属性// 类加载时是线程安全的return StaticObj.OBJ;}// 静态内部类在类加载时不会马上加载,解决懒加载// 只有使用到静态内部类中的属性时,静态内部类才会加载static class StaticObj {private static final Obj OBJ = new Obj();}
}
说明
- 这种方式采用了类装载的机制来保证初始化实例时只有一个线程
- 静态内部类方式在Obj类被装载时不会立即实例化,而是在需要实例化时,调用getObj方法时,才会装载StaticObj类,从而完成Obj的实例化
- 类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的
- 优点;避免了线程不安全,利用静态内部类特点实现懒加载,效率高,
- 结论:推荐使用
枚举方式(推荐方式)
package 单例模式.枚举方式;enum Type {INSTANCE,USER("小韩", 12, "学生");String name;String job;int i;Type(String name, int i, String job) {this.i = i;this.job = job;this.name = name;}Type() {}public void sayOk() {System.out.println("ok");}@Overridepublic String toString() {return "Type{" +"name='" + name + '\'' +", job='" + job + '\'' +", i=" + i +'}';}
}/*** @author Han* @data 2023/10/28* @apiNode*/
public class Test {public static void main(String[] args) {Type instance = Type.INSTANCE;Type instance2 = Type.INSTANCE;Type user1 = Type.USER;Type user2 = Type.USER;System.out.println(Type.INSTANCE);System.out.println(Type.USER);System.out.println(instance == instance2); // trueSystem.out.println(user2 == user1); // true}
}
优点说明
- 借助了JDK1.5中添加的枚举来实现单例模式,不仅能避免多线程问题,而且还能防止反序列化重新创建新的对象
- 这种方式推荐使用
单例模式在JDK中的使用
单例模式注意事项和细节说明
- 单例模式保障了系统中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能
- 当想实例化一个单例对象的使用,必须要记住使用相应的获取对象的方法,而不是使用new
单例模式的使用场景
- 需要频繁的进行创建和销毁对象
- 创建对象是耗时过多或者耗费资源过多,但是又经常使用到的对象,工厂类对象
- 频繁访问数据库或文件的对象(比如数据源,session工厂等)
相关文章:

设计模式——单例模式详解
目录 设计模式类型单例模式单例模式方式饿汉式静态常量方式静态代码块形式 懒汉式线程不安全(不推荐)懒汉式优化(不推荐) 双重检查(推荐方式)静态内部类(推荐方式)枚举方式ÿ…...

一、W5100S/W5500+RP2040树莓派Pico<静态配置网络信息>
文章目录 1. 前言2. 相关网络信息2.1 简介2.2 优点2.3 应用 3. WIZnet以太网芯片4. 静态IP网络设置示例讲解以及使用4.1 程序流程图4.2 测试准备4.3 连接方式4.4 相关代码4.5 编译烧录 5. 注意事项6. 相关链接 1. 前言 从本章开始我们将用WIZnet的W5100S/W5500以太网芯片结合RP…...

【C++的OpenCV】第十四课-OpenCV基础强化(二):访问单通道Mat中的值
🎉🎉🎉 欢迎各位来到小白 p i a o 的学习空间! \color{red}{欢迎各位来到小白piao的学习空间!} 欢迎各位来到小白piao的学习空间!🎉🎉🎉 💖💖&…...

elementUI el-collapse 自定义折叠面板icon 和 样式 或文字展开收起
: :v-deep{.el-collapse-item__arrow {width: 40px;}.el-icon-arrow-right:before {content: "展开";font-size: 15px;font-family: heiti;color: #2295ff;font-weight: bold;}.el-collapse-item__arrow.is-active {transform: none;}.el-collapse-item__arrow.is-a…...

如何用个人数据Milvus Cloud知识库构建 RAG 聊天机器人?(上)
生成式人工智能时代,开发者可以借助大语言模型(LLM)开发更智能的应用程序。然而,由于有限的知识,LLM 非常容易出现幻觉。检索增强生成(RAG)https://zilliz.com/use-cases/llm-retrieval-augmented-generation 通过为 LLM 补充外部知识,有效地解决了这一问题。 在 Chat …...

2023年江西省“振兴杯”工业互联网安全技术技能大赛暨全国大赛江西选拔赛 Write UP
文章目录 一、协议分析 - modbus二、协议分析 - 异常的流量三、协议分析 - S7Error四、协议分析 - OmronAttack五、组态编程 - 工程的秘密六、组态编程 - 工程的秘密七、组态编程 - 简单的计算八、组态编程 - 交通灯九、组态编程 - 有趣的转盘十、应急处置 - 登录日志分析十一、…...

PostMan 之 Mock 接口测试
在测试的时候经常会碰到后端开发工程师的接口还没有开发完成,但是测试任务已经分配过来。没有接口怎么测试呢? 测试人员可以通过 mock server 自己去造一个接口来访问。mock server 可用于模拟真实的接口。收到请求时,它会根据配置返回对应的…...
LuatOS-SOC接口文档(air780E)--libgnss - NMEA数据处理
示例 -- 提醒: 本库输出的坐标,均为 WGS84 坐标系 -- 如需要在国内地图使用, 要转换成对应地图的坐标系, 例如 GCJ02 BD09 -- 相关链接: https://lbsyun.baidu.com/index.php?titlecoordinate -- 相关链接: https://www.openluat.com/GPS-Offset.html-- 方案1, 经lua层进行数…...

基于华为云 IoT 物联网平台实现家居环境实时监控
01 智能家居环境监测 智能家居环境监测采用 Ruff 开发板作为主控,串口线连接温湿度传感器 DHT11 和空气质量传感器 SDS011,每5分钟采集一次数据,通过 MQTT 协议发送到华为云 IoT 物联网平台,并基于数据分析服务实时计算出整个家庭…...

【开源框架】Glide的图片加载流程
本篇文章从Glide 4.11源码入手,简单的分析整个图片请求的流程,本着 ”只见树林,不见树木“ 的原则,宏观请求流程,不细究实现细节(细节留坑埋点,之后慢慢写) 引入依赖 以下的所有分…...
win10下Mariadb绿色版安装步骤
使用绿色版的mariadb数据库管理软件,免费开源,可以用来替换MySQL。首先从mariadb官网下载绿色版本的压缩包。解压后、配置好即可以使用。 把他解压缩到C:\mariadb\之下。打开powershell: Cd c:\mariadb\bin .\mysql_install_db.exe 这一…...

wiresharak捕获DNS
DNS解析: 过滤项输入dns: dns查询报文 应答报文: 事务id相同,flag里 QR字段1,表示响应,answers rrs变成了2. 并且响应报文多了Answers 再具体一点,得到解析出的ip地址(最底下的add…...

vue源码分析(一)——源码目录说明
文章目录 一、如何下载源码(可忽略)(1)打开地址(2)复制链接(3)git clone 链接 二、源码目录说明1.可以根据你下载的源码通过package.json文件查看vue版本2.源码目录说明 一、如何下载…...

【深度学习】吴恩达课程笔记(二)——浅层神经网络、深层神经网络
笔记为自我总结整理的学习笔记,若有错误欢迎指出哟~ 笔记链接 【深度学习】吴恩达课程笔记(一)——深度学习概论、神经网络基础 吴恩达课程笔记——浅层神经网络、深层神经网络 四、浅层神经网络1.双层神经网络表示2.双层神经网络的前向传播第一层前向传播第二层前…...

UI自动化概念 + Web自动化测试框架介绍
1.UI自动化测试概念:我们先明确什么是UI UI,即(User Interface简称UI用户界面)是系统和用户之间进行交互和信息交换的媒介 UI自动化测试: Web自动化测试和移动自动化测试都属于UI自动化测试,UI自动化测试就是借助自动化工具对程序UI层进行自动化的测试 …...
在 macOS 上的多个 PHP 版本之间切换
文章目录 前言一、前提条件1.引入库需要安装 Xcode 2.安装多个PHP版本2.PHP版本切换 开源替代品 前言 不同项目使用php版本可能不同,需要安装不同版本php 一、前提条件 1.引入库 需要安装 Xcode 命令行工具和Homebrew xcode-select --install检查brew是否已安…...
地址解析协议ARP
地址解析协议(Address Resolution Protocol,ARP),用于根据本网内目的主机或默认网关的IP地址获取其MAC地址。 ARP的基本思想:在每一台主机中设置专用内存区域,称为ARP高速缓存(也称为ARP表&…...

Go学习第十三章——Gin入门与路由
Go web框架——Gin入门与路由 1 Gin框架介绍1.1 基础介绍1.2 安装Gin1.3 快速使用 2 路由2.1 基本路由GET请求POST请求 2.2 路由参数2.3 路由分组基本分组带中间件的分组 2.4 重定向 1 Gin框架介绍 github链接:https://github.com/gin-gonic/gin 中文文档…...

[减脂期食谱] 自制千岛酱
[减脂期食谱] 自制千岛酱 成品如下: 最中间的那个,算比较居中的颜色吧,其实自己家做原版的千岛酱还是比较简单的,它的底就是蛋黄酱(蛋黄油乳化的酱),随后里面的材料比较自由,维基百科是这么介绍的…...

Android 系统架构
目录 Android 系统架构 1. Android 应用层 2. Android应用框架层 2.1 Activity Manager (活动管理器) 2.2 Window Manager (窗口管理器) 2.3 Content Provider (内容提供器) 2.4 View System(视图系统&a…...

地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...

树莓派超全系列教程文档--(61)树莓派摄像头高级使用方法
树莓派摄像头高级使用方法 配置通过调谐文件来调整相机行为 使用多个摄像头安装 libcam 和 rpicam-apps依赖关系开发包 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 配置 大多数用例自动工作,无需更改相机配置。但是,一…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...
Java - Mysql数据类型对应
Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作
一、上下文切换 即使单核CPU也可以进行多线程执行代码,CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短,所以CPU会不断地切换线程执行,从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...

智能分布式爬虫的数据处理流水线优化:基于深度强化学习的数据质量控制
在数字化浪潮席卷全球的今天,数据已成为企业和研究机构的核心资产。智能分布式爬虫作为高效的数据采集工具,在大规模数据获取中发挥着关键作用。然而,传统的数据处理流水线在面对复杂多变的网络环境和海量异构数据时,常出现数据质…...

Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...
Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?
Redis 的发布订阅(Pub/Sub)模式与专业的 MQ(Message Queue)如 Kafka、RabbitMQ 进行比较,核心的权衡点在于:简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...