设计模式(四)创建者模式之单例模式

2024-06-19 02:20

本文主要是介绍设计模式(四)创建者模式之单例模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

单例模式

  • 单例设计模式
  • 单例模式的结构
  • 单例模式的实现
    • 饿汉式-方式1(静态变量方式)
    • 饿汉式-方式2(静态代码块方式)
    • 懒汉式-方式1(线程不安全)
    • 懒汉式-方式2(线程安全) synchronized 关键字
    • 懒汉式-方式3(双重检查锁)
    • 懒汉式-方式4(静态内部类方式)
    • 枚举方式
    • 存在的问题
      • 破坏单例模式:
        • 序列化反序列化
        • 反射
    • 问题的解决
      • 序列化、反序列方式破坏单例模式的解决方法
      • 反射方式破解单例的解决方法

创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是“将对象的创建与使用分离”。
这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。

单例设计模式

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

单例模式的结构

单例模式的主要有以下角色:

  • 单例类。只能创建一个实例的类
  • 访问类。使用单例类

单例模式的实现

单例设计模式分类两种:

  • 饿汉式:类加载就会导致该单实例对象被创建
  • 懒汉式:类加载不会导致该单实例对象被创建,而是首次使用该对象时才会创建

饿汉式-方式1(静态变量方式)

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-饿汉式*  静态变量创建类的对象*/
public class SingleDemo {//私有构造方法private    SingleDemo (){}//在成员位置创建该类的对象private static SingleDemo singleDemo=new SingleDemo();//对外提供静态方法获取该对象public static SingleDemo getSingleDemo(){return singleDemo;}
}

说明:
​ 该方式在成员位置声明SingleDemo 类型的静态变量,并创建SingleDemo 类的对象instance。instance对象是随着类的加载而创建的。如果该对象足够大的话,而一直没有使用就会造成内存的浪费

饿汉式-方式2(静态代码块方式)

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-饿汉式*  在静态代码块中创建该类对象*/
public class SingleDemo2 {//私有构造方法private SingleDemo2(){}//在成员位置创建该类的对象private static SingleDemo2 instance ;static  {instance=new SingleDemo2();}//对外提供静态方法获取该对象public static  SingleDemo2 getSingleDemo(){return instance;}
}

说明:
该方式在成员位置声明SingleDemo2 类型的静态变量,而对象的创建是在静态代码块中,也是对着类的加载而创建。所以和饿汉式的方式1(静态变量方式)基本上一样,当然该方式也存在内存浪费问题

懒汉式-方式1(线程不安全)

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-懒汉式* 线程不安全*/
public class SingleDemo3 {//私有构造方法private SingleDemo3() {}//在成员位置创建该类的对象private static SingleDemo3 instance;//对外提供静态方法获取该对象public  static SingleDemo3 getSingleDemo() {if (instance == null) {instance = new SingleDemo3();}return instance;}
}

说明:
从上面代码我们可以看出该方式在成员位置声明SingleDemo3类型的静态变量,并没有进行对象的赋值操作,那么什么时候赋值的呢?当调用getSingleDemo()方法获取SingleDemo3类的对象的时候才创建SingleDemo3类的对象,这样就实现了懒加载的效果。但是,如果是多线程环境,会出现线程安全问题。

懒汉式-方式2(线程安全) synchronized 关键字

package com.lx.design.creator.single;/**
* TODO 添加描述
*
* @author lx
* @date 2024/6/15 12:46
* 单例模式-懒汉式
* 线程 安全
*/
public class SingleDemo4 {//私有构造方法private SingleDemo4() {}//在成员位置创建该类的对象private static SingleDemo4 instance;//对外提供静态方法获取该对象public static synchronized SingleDemo4 getSingleDemo() {if (instance == null) {instance = new SingleDemo4();}return instance;}
}

说明:
该方式也实现了懒加载效果,同时又解决了线程安全问题。但是在getSingleDemo()方法上添加了synchronized关键字,导致该方法的执行效果特别低。从上面代码我们可以看出,其实就是在初始化instance的时候才会出现线程安全问题,一旦初始化完成就不存在了。

懒汉式-方式3(双重检查锁)

再来讨论一下懒汉模式中加锁的问题,对于 getSingleDemo() 方法来说,绝大部分的操作都是读操作,读操作是线程安全的,所以我们没必让每个线程必须持有锁才能调用该方法,我们需要调整加锁的时机。由此也产生了一种新的实现模式:双重检查锁模式

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-懒汉式* 线程 安全*/
public class SingleDemo5 {//私有构造方法private SingleDemo5() {}//在成员位置创建该类的对象private static SingleDemo5 instance;//对外提供静态方法获取该对象public static synchronized SingleDemo5 getSingleDemo() {//第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实例if(instance == null) {synchronized (SingleDemo5.class) {//抢到锁之后再次判断是否为nullif(instance == null) {instance = new SingleDemo5();}}}return instance;}
}

双重检查锁模式是一种非常好的单例实现模式,解决了单例、性能、线程安全问题,上面的双重检测锁模式看上去完美无缺,其实是存在问题,在多线程的情况下,可能会出现空指针问题,出现问题的原因是JVM在实例化对象的时候会进行优化指令重排序操作。

要解决双重检查锁模式带来空指针异常的问题,只需要使用 volatile 关键字, volatile 关键字可以保证可见性和有序性。

更改 instance 成员变量被volatile修饰

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-懒汉式* 线程 安全*/
public class SingleDemo5 {//私有构造方法private SingleDemo5() {}//在成员位置创建该类的对象private static  volatile SingleDemo5 instance;//对外提供静态方法获取该对象public static  SingleDemo5 getSingleDemo() {//第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实例if(instance == null) {synchronized (SingleDemo5.class) {//抢到锁之后再次判断是否为nullif(instance == null) {instance = new SingleDemo5();}}}return instance;}
}

小结:
添加 volatile 关键字之后的双重检查锁模式是一种比较好的单例实现模式,能够保证在多线程的情况下线程安全也不会有性能问题。

懒汉式-方式4(静态内部类方式)

静态内部类单例模式中实例由内部类创建,由于 JVM 在加载外部类的过程中, 是不会加载静态内部类的, 只有内部类的属性或者方法被调用时才会被加载, 并初始化其静态属性。静态属性由于被 static 修饰,保证只被实例化一次,并且严格保证实例化顺序。

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-静态内部类方式*/
public class SingleDemo6 {//私有构造方法private SingleDemo6() {}private static class SingletonHolder {private static final SingleDemo6 INSTANCE = new SingleDemo6();}//对外提供静态方法获取该对象public static SingleDemo6 getInstance() {//访问内部类中的属性或者方法。才会加载内部类。//并初始化其静态属性。静态属性由于被 `static` 修饰,保证只被实例化一次return SingletonHolder.INSTANCE;}
}

说明:
第一次加载SingleDemo6 类时不会去初始化INSTANCE,只有第一次调用getInstance,虚拟机加载SingletonHolder。 并初始化INSTANCE,这样不仅能确保线程安全,也能保证 Singleton 类的唯一性。

小结:
静态内部类单例模式是一种优秀的单例模式,是开源项目中比较常用的一种单例模式。在没有加任何锁的情况下,保证了多线程下的安全,并且没有任何性能影响和空间的浪费。

枚举方式

枚举类实现单例模式是极力推荐的单例实现模式,因为枚举类型线程安全的,并且只会装载一次,设计者充分的利用了枚举的这个特性来实现单例模式,枚举的写法非常简单,而且枚举类型是所用单例实现中唯一一种不会被破坏的单例实现模式。

/*** 枚举方式*/
public enum Singleton {INSTANCE;
}

说明:
​ 枚举方式属于恶汉式方式。

存在的问题

破坏单例模式:

使上面定义的单例类(Singleton)可以创建多个对象,枚举方式除外。有两种方式,分别是序列化和反射。

序列化反序列化
package com.lx.design.creator.single;import java.io.Serializable;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-静态内部类方式*/
public class SingleDemo6 implements Serializable {//私有构造方法private SingleDemo6() {}private static class SingletonHolder {private static final SingleDemo6 INSTANCE = new SingleDemo6();}//对外提供静态方法获取该对象public static SingleDemo6 getInstance() {//访问内部类中的属性或者方法。才会加载内部类。//并初始化其静态属性。静态属性由于被 `static` 修饰,保证只被实例化一次return SingletonHolder.INSTANCE;}
}

测试 Test1 类

package com.lx.design.creator.single;import java.io.*;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-静态内部类方式*/
public class Test1 implements Serializable {public static void main(String[] args) throws Exception {//往文件中写对象writeObject2File();//从文件中读取对象SingleDemo6 s1 = readObjectFromFile();SingleDemo6 s2 = readObjectFromFile();//判断两个反序列化后的对象是否是同一个对象System.out.println(s1 == s2);}private static SingleDemo6 readObjectFromFile() throws Exception {//创建对象输入流对象ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\SingleDemo6.txt"));//第一个读取Singleton对象SingleDemo6 instance = (SingleDemo6) ois.readObject();return instance;}public static void writeObject2File() throws Exception {//获取Singleton类的对象SingleDemo6 instance = SingleDemo6.getInstance();//创建对象输出流ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\SingleDemo6.txt"));//将instance对象写出到文件中oos.writeObject(instance);}}

上面代码运行结果是false,表明序列化和反序列化已经破坏了单例设计模式

反射

**Test2 **

  package com.lx.design.creator.single;import java.io.*;
import java.lang.reflect.Constructor;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-静态内部类方式*/
public class Test2 implements Serializable {public static void main(String[] args) throws Exception {//获取Singleton类的字节码对象Class clazz = SingleDemo5.class;//获取Singleton类的私有无参构造方法对象Constructor constructor = clazz.getDeclaredConstructor();//取消访问检查constructor.setAccessible(true);//创建Singleton类的对象s1SingleDemo5 s1 = (SingleDemo5) constructor.newInstance();//创建Singleton类的对象s2SingleDemo5 s2 = (SingleDemo5) constructor.newInstance();//判断通过反射创建的两个Singleton对象是否是同一个对象System.out.println(s1 == s2);}
}

上面代码运行结果是false,表明序列化和反序列化已经破坏了单例设计模式

注意:枚举方式不会出现这两个问题。

问题的解决

序列化、反序列方式破坏单例模式的解决方法

在Singleton类中添加readResolve()方法,在反序列化时被反射调用,如果定义了这个方法,就返回这个方法的值,如果没有定义,则返回新new出来的对象。

package com.lx.design.creator.single;import java.io.Serializable;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-静态内部类方式*/
public class SingleDemo6 implements Serializable {//私有构造方法private SingleDemo6() {}private static class SingletonHolder {private static final SingleDemo6 INSTANCE = new SingleDemo6();}//对外提供静态方法获取该对象public static SingleDemo6 getInstance() {//访问内部类中的属性或者方法。才会加载内部类。//并初始化其静态属性。静态属性由于被 `static` 修饰,保证只被实例化一次return SingletonHolder.INSTANCE;}/*** 下面是为了解决序列化反序列化破解单例模式*/private Object readResolve() {return SingletonHolder.INSTANCE;}
}

再次执行Test1 发现返回true

反射方式破解单例的解决方法

package com.lx.design.creator.single;/*** TODO 添加描述** @author lx* @date 2024/6/15 12:46* 单例模式-懒汉式* 线程 安全*/
public class SingleDemo5 {//判断是否已经实例化对象标识private static  boolean flag =false;//私有构造方法private SingleDemo5() {//判断是否已经实例化对象if (flag){throw new RuntimeException();}//更改  flag =true;}//在成员位置创建该类的对象private static  volatile SingleDemo5 instance;//对外提供静态方法获取该对象public static   SingleDemo5 getSingleDemo() {//第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实例if(instance == null) {synchronized (SingleDemo5.class) {//抢到锁之后再次判断是否为nullif(instance == null) {instance = new SingleDemo5();}}}return instance;}
}

说明:
这种方式比较好理解。当通过反射方式调用构造方法进行创建创建时,直接抛异常。不运行此中操作。

这篇关于设计模式(四)创建者模式之单例模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1073733

相关文章

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

SpringBoot如何通过Map实现策略模式

《SpringBoot如何通过Map实现策略模式》策略模式是一种行为设计模式,它允许在运行时选择算法的行为,在Spring框架中,我们可以利用@Resource注解和Map集合来优雅地实现策略模式,这... 目录前言底层机制解析Spring的集合类型自动装配@Resource注解的行为实现原理使用直接使用M

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

大数据spark3.5安装部署之local模式详解

《大数据spark3.5安装部署之local模式详解》本文介绍了如何在本地模式下安装和配置Spark,并展示了如何使用SparkShell进行基本的数据处理操作,同时,还介绍了如何通过Spark-su... 目录下载上传解压配置jdk解压配置环境变量启动查看交互操作命令行提交应用spark,一个数据处理框架

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试