当前位置: 首页 > news >正文

设计模式——单例模式详解

目录

  • 设计模式类型
  • 单例模式
    • 单例模式方式
      • 饿汉式
        • 静态常量方式
        • 静态代码块形式
      • 懒汉式
        • 线程不安全(不推荐)
        • 懒汉式优化(不推荐)
      • 双重检查(推荐方式)
      • 静态内部类(推荐方式)
      • 枚举方式(推荐方式)
  • 单例模式在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中的使用

image-20231028131029096

单例模式注意事项和细节说明

  • 单例模式保障了系统中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能
  • 当想实例化一个单例对象的使用,必须要记住使用相应的获取对象的方法,而不是使用new

单例模式的使用场景

  • 需要频繁的进行创建和销毁对象
  • 创建对象是耗时过多或者耗费资源过多,但是又经常使用到的对象,工厂类对象
  • 频繁访问数据库或文件的对象(比如数据源,session工厂等)

相关文章:

设计模式——单例模式详解

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

一、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 中文文档&#xf…...

[减脂期食谱] 自制千岛酱

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

Android 系统架构

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

React 第五十五节 Router 中 useAsyncError的使用详解

前言 useAsyncError 是 React Router v6.4 引入的一个钩子,用于处理异步操作(如数据加载)中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误:捕获在 loader 或 action 中发生的异步错误替…...

(十)学生端搭建

本次旨在将之前的已完成的部分功能进行拼装到学生端,同时完善学生端的构建。本次工作主要包括: 1.学生端整体界面布局 2.模拟考场与部分个人画像流程的串联 3.整体学生端逻辑 一、学生端 在主界面可以选择自己的用户角色 选择学生则进入学生登录界面…...

Xshell远程连接Kali(默认 | 私钥)Note版

前言:xshell远程连接,私钥连接和常规默认连接 任务一 开启ssh服务 service ssh status //查看ssh服务状态 service ssh start //开启ssh服务 update-rc.d ssh enable //开启自启动ssh服务 任务二 修改配置文件 vi /etc/ssh/ssh_config //第一…...

IGP(Interior Gateway Protocol,内部网关协议)

IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...

《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)

CSI-2 协议详细解析 (一) 1. CSI-2层定义(CSI-2 Layer Definitions) 分层结构 :CSI-2协议分为6层: 物理层(PHY Layer) : 定义电气特性、时钟机制和传输介质(导线&#…...

CMake基础:构建流程详解

目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端

🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》

在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中&#xff0…...

家政维修平台实战20:权限设计

目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色&#xf…...

Python实现prophet 理论及参数优化

文章目录 Prophet理论及模型参数介绍Python代码完整实现prophet 添加外部数据进行模型优化 之前初步学习prophet的时候,写过一篇简单实现,后期随着对该模型的深入研究,本次记录涉及到prophet 的公式以及参数调优,从公式可以更直观…...