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

目录

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

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

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

相关文章

STM32 ADC数模转换器

STM32 ADC数模转换器 ADC简介 ADC(Analog-Digital Converter)模拟-数字转换器 ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁 STM32主要是数字电路,数字电路只有高低电平&#xf…

【torch高级】一种新型的概率学语言pyro(01/2)

一、说明 贝叶斯推理,也就是变分概率模型估计,属于高级概率学模型,极有学习价值;一般来说,配合实际活动学习可能更直观,而pyro是pytorch的概率工具,不同于以往的概率工具,只是集中于…

qt中怎么在鼠标停留的位置上显示该点的坐标位置

需要重写控件的mouseMoveEvent方法。 1、自定义一个QLabel控件&#xff0c;然后重写QLabel的mouseMoveEvent customlabel.h#include <QWidget> #include <QHBoxLayout> #include <QLabel>class CustomLabel : public QLabel {Q_OBJECT public:explicit Cus…

python常见爬虫库以及案例

python常见爬虫库以及案例 一、常见库 以下是一些常见的Python爬虫库&#xff0c;按照一般热门程度的排序&#xff1a; Requests&#xff1a;requests库是非常流行的用于发送HTTP请求的库&#xff0c;因其简洁易用和广泛的社区支持而备受青睐。Beautiful Soup&#xff1a;Be…

PY32F002A系列单片机:高性价比、低功耗,满足多样化应用需求

PY32F002A系列微控制器是一款高性能、低功耗的MCU&#xff0c;它采用32位ARM Cortex-M0内核&#xff0c;最高工作频率达到24MHz&#xff0c;提供了强大的计算能力。此外&#xff0c;PY32F002A拥有最大20Kbytes的flash存储器和3Kbytes的SRAM&#xff0c;为简单的数据处理提供了充…

基于LCC的Buck谐振变换器研究

摘 要 Buck 变换器应用广泛&#xff0c;比如可以为音圈电机、直流电机以及电子设备等提供直流供电电源。更高效率和更小体积的Buck 直流调压电源一直是研究的热点。在我们日常生活中&#xff0c; LLC谐振变换器和Buck谐振变换器随处可见&#xff0c;因为其相比其他变换器而言结…

Python Selenium 之数据驱动测试的实现!

数据驱动模式的测试好处相比普通模式的测试就显而易见了吧&#xff01;使用数据驱动的模式&#xff0c;可以根据业务分解测试数据&#xff0c;只需定义变量&#xff0c;使用外部或者自定义的数据使其参数化&#xff0c;从而避免了使用之前测试脚本中固定的数据。可以将测试脚本…

Unity3D 如何用unity引擎然后用c#语言搭建自己的服务器

Unity3D是一款强大的游戏开发引擎&#xff0c;可以用于创建各种类型的游戏。在游戏开发过程中&#xff0c;经常需要与服务器进行通信来实现一些功能&#xff0c;比如保存和加载游戏数据、实现多人游戏等。本文将介绍如何使用Unity引擎和C#语言搭建自己的服务器&#xff0c;并给…

thrust工程化学习(七)----噪声滤除进阶版

0. 简介 之前我们讲过通过体素化分割&#xff0c;并通过判断这个栅格内的点云数目是否大于阈值。从而来鉴别出噪点。而我们学过最近邻搜索后&#xff0c;我们可以来学习一下更加先进的方法—半径搜索噪声滤除&#xff08;Radius Search Noise Filtering&#xff09;。这是点云…

Redis(05)| 数据结构-哈希表

哈希表是一种保存键值对&#xff08;key-value&#xff09;的数据结构。 哈希表中的每一个 key 都是独一无二的&#xff0c;程序可以根据 key 查找到与之关联的 value&#xff0c;或者通过 key 来更新 value&#xff0c;又或者根据 key 来删除整个 key-value等等。 在讲压缩列表…

报错:Could not resolve host: mirrorlist.centos.org;Unknown error

报错&#xff1a;Could not resolve host: mirrorlist.centos.org;Unknown error 一般是因为网络配置错误导致无法连接外网&#xff0c;我们先尝试ping一下www.baidu.com发现无法ping通。 果然&#xff0c;接下来我们就开始排查吧&#xff01;&#xff01; 1.网络配置查看 打开…

SpringBoot解压zip包,读取每个文件内容

SpringBoot解压zip包&#xff0c;读取每个文件内容 一、运用场景 获取本地压缩包&#xff0c;解压后根据文件名称及类型&#xff0c;对读取的文件内容进行业务处理。 二、POM文件依赖 <!--读取文件--><dependency><groupId>org.apache.poi</groupId&g…

win10下Mariadb绿色版安装步骤

使用绿色版的mariadb数据库管理软件&#xff0c;免费开源&#xff0c;可以用来替换MySQL。首先从mariadb官网下载绿色版本的压缩包。解压后、配置好即可以使用。 把他解压缩到C:\mariadb\之下。打开powershell&#xff1a; Cd c:\mariadb\bin .\mysql_install_db.exe 这一…

vue制作防止用户未登录或未填写信息就跳转页面的路由拦截器

在Vue中&#xff0c;你可以使用路由导航守卫来实现防止未登录用户跳转页面的路由拦截器。 首先&#xff0c;你需要创建一个全局前置守卫&#xff0c;用于检查用户是否已登录。在router/index.js文件中&#xff0c;添加如下代码&#xff1a; import router from /router;route…

MySQL8锁的问题

关键字 mysql 8、lock 问题描述 项目上反馈&#xff0c;一个简单的提交操作需要 40 秒。 抓取 SQL 发现 update gl_credit_bill set verifystate2 where id2761279790403840 执行耗时近40秒解决问题思路 手动执行 SQL&#xff0c;发现非常快&#xff0c;基本排除数据库本身…

@CallSuper注解方法学习

CallSuper注解是什么&#xff1f; CallSuper 是 Android 开发中使用的一个注解&#xff0c;它的主要用途是确保在子类重写父类的方法时&#xff0c;调用 super 方法。这在某些情况下是非常有用的&#xff0c;例如当你希望在重写方法时保留父类的默认行为&#xff0c;或者确保子…

24年FRM备考知识点以及一级公式表

FRM一级公示表以及备考知识点 链接&#xff1a;https://pan.baidu.com/s/17RpFF9OyfRk7FGtEQrxf3A?pwd1234 提取码&#xff1a;1234 FRM二级公示表以及备考知识点 链接&#xff1a;https://pan.baidu.com/s/175D05wV1p94dIfBZThutCQ?pwd1234 提取码&#xff1a;1234

HarmonyOS原生分析能力,即开即用助力精细化运营

数据分析产品对开发者的价值呈现在两个层面&#xff0c;第一个是产品的层面&#xff0c;可以通过数据去洞察用户的行为&#xff0c;从而找到产品的优化点。另外一个就是运营层面&#xff0c;可以基于数据去驱动&#xff0c;来实现私域和公域的精细化运营。 在鸿蒙生态上&#…

conda 实践

1. 环境部署 1.1. 下载 anaconda 安装包 下面这个网址查找自己需要的版本 https://repo.anaconda.com/archive/ 或者手动下载。 wget https://repo.anaconda.com/archive/Anaconda3-5.3.0-Linux-x86_64.sh 1.2. 执行安装程序 #安装依赖&#xff1a; sudo yum install bzip2…

APP自动化测试 ---- Appium介绍及运行原理

在面试APP自动化时&#xff0c;有的面试官可能会问Appium的运行原理&#xff0c;以下介绍Appium运行原理。 一、Appium介绍 1.Appium概念 Appium是一个开源测试自动化框架&#xff0c;可用于原生&#xff0c;混合和移动Web应用程序测试。它使用WebDriver协议驱动IOS&#xf…