一、单例模式(Singleton Pattern)

一、概述

单例模式是指在整个软件系统中,一个类只有一个实例对象,并且该类需要提供访问该实例对象的方法(静态方法)

二、UML图

在这里插入图片描述

三、适用场景

  1. 当需要频繁的创建和销毁对象,并且创建对象耗时较长,资源消耗较多,但是会经常使用到的对象,就需要使用单例模式

四、特点

  1. 一个类只有一个实例

五、单例模式的7种实现形式

1)饿汉式(静态常量)

  • 示例
public class Singleton_01 {

    private static final Singleton_01 instance = new Singleton_01();

    private Singleton_01(){}

    public static Singleton_01 getInstance() {
        return instance;
    }
}

class Test_01 {

    public static void main(String[] args){
        Singleton_01 instance1 = Singleton_01.getInstance();
        Singleton_01 instance2 = Singleton_01.getInstance();
        System.out.println(instance1 == instance2);
        System.out.println("instance1 hash code: " + instance1.hashCode());
        System.out.println("instance1 hash code: " + instance2.hashCode());
    }
}
  • 优缺点说明
  1. 优点:写法简单,在类加载完成后就完成了初始化,避免了线程同步的问题
  2. 缺点:在类加载完成后完成了实例化,没有达到Lazy Loading的效果,如果系统从始至终没有使用过该实例,便造成了内存的浪费
  3. 说明:这种方式基于ClassLoader机制(ClassLoader机制本身是线程安全的)避免了多线程的同步问题,但是,其实例会在类加载完成之后立即完成了初始化;如果由于其他原因,导致了该类的加载,便会导致该类的实例被初始化,没有达到Lazy Loading的效果
  4. 结论:写法简单,但是可能会造成内存浪费;如果确定实例会使用,建议使用

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

  • 示例
public class Singleton_02 {

    private static Singleton_02 instance;

    static {
        instance  = new Singleton_02();
    }

    private Singleton_02(){}

    public static Singleton_02 getInstance() {
        return instance;
    }
}

class Test_02 {

    public static void main(String[] args){
        Singleton_02 instance1 = Singleton_02.getInstance();
        Singleton_02 instance2 = Singleton_02.getInstance();
        System.out.println(instance1 == instance2);
        System.out.println("instance1 hash code: " + instance1.hashCode());
        System.out.println("instance1 hash code: " + instance2.hashCode());
    }
}
  • 优缺点说明
  1. 同饿汉式静态常量的方式一样,只不过把示例的初始化过程放在了静态代码块中;

3)懒汉式(线程不安全)

  • 示例
public class Singleton {

    private static Singleton instance;

    private Singleton(){}

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

class Test_02 {

    public static void main(String[] args){
        // 多线程下测试线程不安全问题(执行多次, 会发现输出的hashCode不同)
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                Singleton instance = Singleton.getInstance();
                System.out.println(instance.hashCode());
            }, "T-" + i).start();
        }
    }
}
  • 优缺点说明
  1. 起到了Lazy Loading的效果,但是线程不安全
  2. 在多线程环境下,两个线程同时执行到if (instance == null),若为true,则会创建两个不同的实例
  3. 多线程环境下不能使用该形式去创建单例

4)懒汉式(线程安全,同步方法)

  • 示例
public class Singleton {

    private static Singleton instance;

    private Singleton(){}

    public synchronized static Singleton getInstance() {
        if (instance == null)
            instance = new Singleton();
        return instance;
    }
}

class Test {

    public static void main(String[] args){

        // 多线程下测试线程不安全问题(执行多次, 线程安全)
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                Singleton instance = Singleton.getInstance();
                System.out.println(instance.hashCode());
            }, "T-" + i).start();
        }
    }
}
  • 优缺点说明
  1. 使用synchronized修饰静态方法, 保证了线程安全
  2. 但是每次调用getInstance方法时, 都需要获取锁, 增加等待时间, 效率较低

5)双重检查

public class Singleton {

    private volatile static Singleton instance;

    private Singleton(){}

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null){
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

class Test {

    public static void main(String[] args){

        // 多线程下测试线程不安全问题(执行多次, 线程安全)
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                Singleton instance = Singleton.getInstance();
                System.out.println(instance.hashCode());
            }, "T-" + i).start();
        }
    }
}
  • 优缺点说明
  1. Double-Check概念是多线程开发中常用的方式;示例中进行了两次if(instance == null)的检查,第一次检查是为了在创建对象时,判断对象是否为空,若为空则进行创建对象;第二次检查,并且加了同步锁,为了避免多线程情况下,两个线程同时通过了第一次检查,都进入了创建对象的步骤,第二次加锁检查,就只能允许一个线程进行对象的创建
  2. 确保了实例化代码只需要执行一次,后面再次访问时,直接返回实例化之后的对象
  3. 线程安全,延迟加载,效率较高;实际开发中推荐使用

6)静态内部类

public class Singleton {

    private volatile static Singleton instance;

    private Singleton(){}

    public static Singleton getInstance() {
        return InnerSingleton.INSTANCE;
    }

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

class Test {

    public static void main(String[] args){

        // 多线程下测试线程不安全问题(执行多次, 线程安全)
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                Singleton instance = Singleton.getInstance();
                System.out.println(instance.hashCode());
            }, "T-" + i).start();
        }
    }
}
  • 有缺点说明
  1. 采用了类装载的机制保证了线程安全
  2. 静态内部类InnerSingleton在Singleton类被加载时不会立即被初始化,而是在调用getInstance方法时才会对InnerSingleton进行加载操作,加载InnerSingleton会对静态常量进行初始化操作
  3. 利用静态内部类延迟加载的特性实现了懒加载,效率较高

7)枚举

  • 示例
public class Test {

    public static void main(String[] args){
        Singleton instance = Singleton.INSTANCE;
        instance.method();
    }
}

enum Singleton {
    INSTANCE;

    public void method() {
        System.out.println("Do Something!");
    }
}
  • 优缺点说明
  1. 枚举保证了线程安全
  2. 枚举可以避免反序列化时重复创建对象

六、spring中的单例

1、spring中单例bean的创建流程

1.1 注册初始的BeanDefinition,此时scope属性为空
  1. 调用DefaultListableBeanFactory#registerBeanDefinition方法
  2. 此时只是初始化了部分BeanDefinition(Spring的基础Bean,及main方法所在的Bean)
1.2 初始化scope
  1. 调用AbstractBeanFactory#getMergedBeanDefinition,将scope为空的BeanDefinition设置为singleton
if (!StringUtils.hasLength(mbd.getScope())) {
	mbd.setScope(SCOPE_SINGLETON);
}
1.3 注册业务Bean·

七、单例模式的破坏

1、反射的方式破坏单例

☞ 原理

通过反射的方式,获取到私有的构造器,设置可以访问私有构造器的权限,然后通过反射调用的方式去创建实例

public class Reflect {

    public static void main(String[] args) throws Exception {
        Class<?> clazz = Class.forName("com.lizza.d1_singleton.singleton_1.Singleton");
        Constructor<?> constructor = clazz.getDeclaredConstructor();
        constructor.setAccessible(true);
        Singleton instance_1 = (Singleton) constructor.newInstance();
        Singleton instance_2 = (Singleton) constructor.newInstance();
        System.out.println(instance_1.hashCode());
        System.out.println(instance_2.hashCode());
    }
}
☞ 如何避免

在私有构造器中进行判空处理

private Singleton(){
    if (instance != null) {
        throw new RuntimeException("instance is must be null!");
    }
}

2、序列化和反序列化破坏单例

☞ 原理

通过对象序列化可以得到不同的克隆对象,以此来破坏单例

public class Serialize {

    public static void main(String[] args){
        // 传统单例
        com.lizza.d1_singleton.singleton_1.Singleton instance = com.lizza.d1_singleton.singleton_1.Singleton.getInstance();
        // 枚举: 可以避免序列化破坏单例的情况
//        com.lizza.d1_singleton.singleton_7.Singleton instance = com.lizza.d1_singleton.singleton_7.Singleton.INSTANCE;
        Object object = clone(instance);
        System.out.println(instance.getClass() + ", " + instance.hashCode());
        System.out.println(instance.getClass() + ", " + object.hashCode());
    }

    public static Object clone(Object object) {
        ObjectInputStream is = null;
        ObjectOutputStream os = null;
        try {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            os = new ObjectOutputStream(bos);
            os.writeObject(object);
            is = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
            
            return is.readObject();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (os != null) {
                    os.close();
                }
                if (is != null) {
                    is.close();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return null;
    }
}
☞ 解决方式

在Singleton中增加readResolve方法,反序列化时会判断是否有readResolve方法, 若有则通过该方法创建对象

/** 反序列化时会判断是否有readResolve方法, 若有则通过该方法创建对象 **/
private Object readResolve() {
    System.out.println(">>> readResolve is run!");
    return getInstance();
}

参考

【1】破坏单例模式
【2】为什么推荐使用枚举来创建单例
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值