单例模式

简介
  • 这一模式的目的是使得类的一个对象成为系统中的唯一实例
    -要实现这一点,可以从客户端对其进行实例化开始。因此需要用一种只允许生成对象类的唯一实例的机制,“阻止”所有想要生成对象的访问。使用工厂方法来限制实例化过程。
  • 这个方法应该是静态方法(类方法),因为让类的实例去生成另一个唯一实例毫无意义。
  1. 一是某个类只能有一个实例;
  2. 二是它必须自行创建这个实例;
  3. 三是它必须自行向整个系统提供这个实例。

饱汉型:

pubic class Singleton {
    private static Singleton instance = new Singleton();
    private Singleton() {}
    public static Singleton getInstance() {
        return instance;
    }
}

对象在类中被定义为private static,通过getInstance(),通过java的classLoader机制保证了单例对象唯一。
Singleton类在类加载时就被初始化,对instance进行实例化。
天生线程安全,类加载机制保证

饿汉型:

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

实现了延迟初始化
但需要synchronized来保证线程安全。

dcl双重检查:

public class Singleton {
    private static volatile Singleton instance = null;
    private Singleton() {}
    public static Singleton getInstance() {
            // here1
        if (instance == null) {
                // here
            synchronized(Singleton.class) {
                if (instance == null)
                    instance = new Singleton();  // 
            }
        }
        return instance;
    }

}

第一个if (instance == null) 是因为大多数情况下 instance != null
第二个if (instance == null) 是因为可能会存在多个线程执行到here,
例如A,B线程都到了here,然后A先向下执行,new一个Singleton对象,
A执行完后,B向下执行,如果没有if (instance == null)判断B线程也会直接new一个Singleton对象。
就不是单例了。


instance使用volatile修饰的原因:

instance = new Singleton(); 这一行代码存在3个步骤

memory = allocate();    //1,分配对象的内存空间
ctorInstance(memory);   //2,初始化对象
instance = memory;      //3,设置instance指向刚分配的内存地址

23之间可能会被重排序。


memory = allocate();    //1,分配对象的内存空间
instance = memory;      //3,设置instance指向刚分配的内存地址
                        // 注意,此时对象还没有被初始化!
ctorInstance(memory);   //2,初始化对象

重排序在单线程情况下不会出现问题,但是在多线程情况下,会出现问题

比如A线程执行到here1,B线程执行到instance = new Singleton();
因为指令重排序,执行到了3,还没执行2,这时instance != null A线程往下走,
会直接return instance,但此时的instance还没初始化。


将instance用volatile修饰后,通过内存屏障,23的指令重排序会被禁止。

内部类实现线程安全的懒加载

public class Singleton {

    private Singleton(){}
    
    private static class InstanceHolder {
        public static Singleton instance = new Singleton();
    }

    public static Singleton getInstance() {
        return InstanceHolder.instance;
    }
}

利用类加载实现 懒加载

只要应用中不使用内部类,JVM就不会去加载这个单例类,也就不会创建单例对象,
从而实现懒汉式的延迟加载。也就是说这种方式可以同时保证延迟加载和线程安全

枚举类实现单列模式

好处: 能够 防止 反射,序列化,克隆 对单列模式的破环


public enum SingletonClass {
    INSTANCE;

    public void doSomeThing() {

    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值