【创建者模式】单例模式

本文详细介绍了Java中单例设计模式的实现,包括饿汉式、懒汉式(线程不安全、Synchronized、双重检查锁、volatile修饰和静态内部类实现),并讨论了单例模式可能被破坏的情况,如序列化和反序列化以及反射。同时,提供了防止单例破坏的解决方案。
摘要由CSDN通过智能技术生成

1、简介

所谓类的单例设计模式 ( singleton ),就是采取了一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的静态方法。由于单例模式只生成了一个实例,减少了系统性能开销,当一个对象的产生需要比较多的资源时,如读取配置、产生其他依赖对象时,则可以通过在应用启动时直接产生一个单例对象,然后永久驻留内存的方式来解决。

2、构建思想

如果我们要让类在一个虚拟机中只能产生一个对象,我们首先必须将类的构造器的访问权限设置成 private ,这样就不能通过 new 操作符在累的外部产生类的对象了,但在类内部仍可以产生该类的对象。又因为在类的外部开始还无法得到类的对象,只能调用该类的某个静态方法以返回类内部创建的对象,静态方法只能访问类中的静态成员变量,所以,指向类内部产生的该类对象的变量也必须定义成静态的。我将其总结成以下几个步骤:

  1. 将类的构造器的访问权限设置成 private
  2. 将指向类内部的该类对象的变量定义成 static 静态的
  3. 将获取该对象实例方法定义成 static 静态的

3、创建方式

单例设计模式分类两种:

饿汉式:类加载就会导致该单实例对象被创建

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

3.0、固定测试类

在后续创建方式中,如果没有单独编写测试类,则统一使用这一测试类

public class SingletonText1 {
    public static void main(String[] args) {
        Singleton singleton1 = Singleton.getInstance();
        Singleton singleton2 = Singleton.getInstance();
        System.out.println(singleton1 == singleton2);     //控制台打印true,则证明两个是同一个对象
    }
}

3.1、饿汉式

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

class Singleton {

    //1. 私有化类的构造器
    private Singleton() {

    }
    //2. 内部创建类的对象,并且需要设置成 static,因为静态方法中只能调用 static 属性,并且需要控制只有一个对象
    private static Singleton instance = new Singleton();

    //3. 由于构造器进行了私有化不能直接 new, 因此需要提供公共的静态的方法,返回类的对象
    public static Singleton getInstance() {
        return instance;
    }
}

3.2、懒汉式-线程不安全

该方式在成员位置声明Singleton类型的静态变量,并没有进行对象的赋值操作,当调用getInstance()方法获取Singleton类的对象的时候才创建Singleton类的对象,这样就实现了懒加载的效果。但是,如果是多线程环境,会出现线程安全问题

public class Singleton {
    // 私有构造方法
    private Singleton() {}

    // 在成员位置创建该类的对象
    private static Singleton instance;

    //对外提供静态方法获取该对象
    public static Singleton getInstance() {

        if(instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

3.3、懒汉式-Synchronize

该方式也实现了懒加载效果,同时又解决了线程安全问题。但是在getInstance()方法上添加了synchronized关键字,导致该方法的执行效果特别低。与此同时,这一方法并不是一定确保了线程安全,在初始化instance的时候依旧可能会出现线程安全问题,一旦初始化完成就不存在了。

public class Singleton {
    // 私有构造方法
    private Singleton() {}

    // 在成员位置创建该类的对象
    private static Singleton instance;

    // 对外提供静态方法获取该对象
    public static synchronized Singleton getInstance() {
		// 判断是否已经实例化
        if(instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

3.4、懒汉式-双重检查锁

使用双重检查锁机制后运行顺序就成了:

  1. 检查变量是否被初始化(不去争夺锁),如果已被初始化则立即返回。
  2. 获取锁。
  3. 再次检查变量是否已经被初始化,如果还没被初始化就初始化一个对象。

执行双重检查是因为,如果多个线程同时了通过了第一次检查,并且其中一个线程首先通过了第二次检查并实例化了对象,那么剩余通过了第一次检查的线程就不会再去实例化对象。

这样,除了初始化的时候会出现加锁的情况,后续的所有调用都会避免加锁而直接返回,解决了性能消耗的问题。

public class Singleton { 

    // 私有构造方法
    private Singleton() {}

    private static Singleton instance;

   // 对外提供静态方法获取该对象
    public static Singleton getInstance() {
		// 第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实例
        if(instance == null) {
            synchronized (Singleton.class) {
                // 抢到锁之后再次判断是否为null
                if(instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

3.5、懒汉式-volatile

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

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

public class Singleton {

    //私有构造方法
    private Singleton() {}

    private static volatile Singleton instance;

   //对外提供静态方法获取该对象
    public static Singleton getInstance() {
		//第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实际
        if(instance == null) {
            synchronized (Singleton.class) {
                //抢到锁之后再次判断是否为空
                if(instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

3.6、懒汉式-静态内部类

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

第一次加载Singleton类时不会去初始化INSTANCE,只有第一次调用getInstance,虚拟机加载SingletonHolder初始化INSTANCE。在没有加任何锁的情况下,保证了多线程下的安全,并且没有任何性能影响和空间的浪费。

public class Singleton {

    //私有构造方法
    private Singleton() {}

    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

4、单例破坏

虽然在上面的介绍中都有对单例模式的创建方式及改进做了一个介绍,但是按照目前的版本任然存在问题,一些特殊的场景下可能会对单例进行破坏。以下两种情况可导致单例破坏:

  • 序列化/反序列化;
  • 反射。

4.1、序列化和反序列化

序列化意义是将实现序列化的Java对象转换成字节序列 ,这些字节序列可以被保存在磁盘上,或者通过网络传输。以备以后重新恢复成原来的对象。

对于单例类使用序列化、反序列化操作时,会破坏单例(序列化前的对象和反序列化后得到的对象内存地址不同)

4.1.1、破坏演示
package top.xbaoziplus.singleton;

import java.io.*;

public class Singleton implements Serializable {

    //私有构造方法
    private Singleton() {}

    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }

    /**
     * 测试
     */
    public static void main(String[] args) throws Exception {
        // 往文件中写对象
        writeObject2File();
        // 从文件中读取对象
        Singleton s1 = readObjectFromFile();
        Singleton s2 = readObjectFromFile();

        //判断两个反序列化后的对象是否是同一个对象
        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s1 == s2);
    }

    /**
     * 从文件中反序列化成对象
     */
    private static Singleton readObjectFromFile() throws Exception {
        // 创建对象输入流对象
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\a.txt"));
        // 第一个读取Singleton对象
        Singleton instance = (Singleton) ois.readObject();

        return instance;
    }

    /**
     * 序列化到文件
     */
    public static void writeObject2File() throws Exception {
        // 获取Singleton类的对象
        Singleton instance = Singleton.getInstance();
        // 创建对象输出流
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\a.txt"));
        // 将instance对象写出到文件中
        oos.writeObject(instance);
    }
}

运行结果显示通过序列化前后对象不同,表明单例已经被破坏了

top.xbaoziplus.singleton.Singleton@5c7fa833
top.xbaoziplus.singleton.Singleton@39aeed2f
false
4.1.2、原因分析

在学习过程中其实并没有接触到这个原因的分析,这一块空白自己好像又不太心甘的样子,因此,配合上资料加查看源码,大致弄清了其原因,以下便针对源码进行分析。从代码中的读取输入流方法readObject() 入手,逐个查看其底层源码。
image-20221018213310266

一路点击readObject()方法直到 readObject0() 方法,点击进去

image-20221018213552449

找到switch通道中的对象入口,返回值会调用readOrdinaryObject方法,点进去 readOrdinaryObject()

image-20221018213811572

方法中通过三目允许算符判断了对象是否可实例化,如果是可实例化的会通过newInstance()方法反射实例化一个新的对象,所以序列化前的对象和反序列化后得到的对象不同

image-20221018214228487

这时问题出现的原因就找到了,序列化/反序列化导致单例破坏的原因竟是因为其底层使用到了反射,而反射则是另外一个造成单例破坏的原因之一。接下来将针对反射问题作出对应的解决方案。

4.2、反射

4.2.1、破坏演示
public class SingletonTest {
    public static void main(String[] args) throws Exception {
        //获取Singleton类的字节码对象
        Class clazz = Singleton.class;
        //获取Singleton类的私有无参构造方法对象
        Constructor constructor = clazz.getDeclaredConstructor();
        //取消访问检查
        constructor.setAccessible(true);

        //创建Singleton类的对象s1
        Singleton s1 = (Singleton) constructor.newInstance();
        //创建Singleton类的对象s2
        Singleton s2 = (Singleton) constructor.newInstance();

        //判断通过反射创建的两个Singleton对象是否是同一个对象
        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s1 == s2);
    }
}

运行结果显示通过序列化前后对象不同,表明单例已经被破坏了

top.xbaoziplus.singleton.Singleton@6a5fc7f7
top.xbaoziplus.singleton.Singleton@3b6eb2ec
false
4.2.2、问题解决

重新根据4.1.2中的源码分析,往下翻的时候我们可以看到下面这么一段代码,在通过三目运算创建了对象之后,还会去找这个对象里是否有readResolve()方法,如果有,则通过这方法返回对象。
image-20221018215801686

这样问题的解决方案就很明确了,既然会判断是否有readResolve()方法,那么我们主动在需要单例的类中创建这一个方法,并在该方法中获取对象实例不就好了。这里需要注意的是这一方法需要返回的数据类型为 Object 类型;

public class Singleton implements Serializable {

    //私有构造方法
    private Singleton() {}

    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }

    /**
     * 解决反射破坏单例问题
     */
    private Object readResolve() {
        return getInstance();
    }
}
4.2.3、多说一句

这里需要注意的是,在4.2.1中的破坏演示中是直接获取其无参构造函数进行生成对象的,这种情况是无法自动判断是否存在readResolve()方法的,当然,我们可以模拟源码中ObjectStreamClass desc = readClassDesc(false);的方式一步步判断从而执行readResolve()方法。

但这种方法毕竟治标不治本,因此我们可以通过抛异常的方式来提示用户不允许破坏规则生成多个对象实例。

注意的是在构造器中不需要加锁甚至双重检查锁,因为这是懒汉单例,在getInstance()方法中我们已经写好的双重检查锁已经帮我们把关了,避免了线程安全问题,无需重复加锁,做无用功的同时还影响了性能和资源消耗。

public class Singleton {

    //私有构造方法
    private Singleton() {
        /*
           反射破解单例模式需要添加的代码
        */
        if(instance != null) {
            throw new RuntimeException();
        }
    }
    
    private static volatile Singleton instance;

    //对外提供静态方法获取该对象
    public static Singleton getInstance() {

        if(instance != null) {
            return instance;
        }

        synchronized (Singleton.class) {
            if(instance != null) {
                return instance;
            }
            instance = new Singleton();
            return instance;
        }
    }
}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陈宝子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值