单例设计模式

单例模式(Singleton Pattern)是一种设计模式,它的目的是确保一个类在整个程序的生命周期内只有一个实例,并且提供全局访问点来获取该实例。

实现方式

懒加载 - 懒汉模式
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;
    }
}

 这里面用了双重检查锁实现了线程安全的单例模式 ,第一次检查看 instance 是否为空,如果为空,就进行第二次检查。在两次检查之间我对 Singletin.class 加锁,这是个类对象,在类加载的时候会生成,是唯一的。

所以如果有多个线程都通过了第一次检查,那么在这就保证只有一个线程可以进入同步块。进入之后再进行一次检查,因为这个时候只有一个线程在同步块,所以检查判定成功的话,就创建实例,而且只会创建一个,不会有并发问题。

使用双重检查所可以保证在线程安全前提下去创建对象,保证单例。

使用synchronized关键字会有一定的性能开销,但是通过双重检查锁定,只在必要时才进行同步,减少了性能损失。

预加载 - 饿汉模式
public class Singleton {
    private static final Singleton INSTANCE = new Singleton();

    private Singleton() {}

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

 预加载模式是在类加载的时候就创建了单例实例。在预加载模式下,Singleton 类被加载时,静态成员 INSTANCE 会被立即初始化,实例会在类加载的同时创建。

静态内部类模式

在这种实现中,静态内部类的实例是在运行时第一次被访问时创建的,而不是在编译时或类被加载时。

public class Singleton {
    private Singleton() {}

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

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

Singleton 类被加载时,静态内部类 SingletonHolder 不会被立即加载和初始化。

只有当 getInstance() 方法第一次被调用时,SingletonHolder 类才会被加载和初始化,此时会创建 INSTANCE 对象。

枚举模式
public enum Singleton {
    INSTANCE;

    public void doSomething() {
        // 业务逻辑
    }
}

枚举模式是实现单例的最佳方法

枚举模式被认为是实现单例的最佳方法,因为它具备以下特点:

  • 线程安全:枚举实例在类加载时由JVM自动初始化,天然线程安全,无需额外同步处理。
  • 防止反射攻击:枚举的构造器是私有的,Java反射机制无法调用,避免了反射破坏单例。
  • 防止序列化破坏:枚举类型在序列化和反序列化过程中,由JVM保证实例的唯一性,不会生成新的实例。

应用场景

饿汉式:适合资源消耗低且实例需要频繁使用的场景。因为无论实例用不用都会被先加载出来。

双重检查加锁--懒汉式:适合需要延迟加载的场景。

静态内部类:适合需要延迟加载且追求简单高效的多线程环境。

枚举式:适合要求高安全性(防反射、防序列化破坏)且实现简洁的场景。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值