butterfly

单例模式

2019-07-16  本文已影响0人  rbty

单例模式

单例模式的实现方式

饿汉式(静态常量)

注:静态常量位于虚拟机内存的方法区,是线程共享的。

public class Singleton {

    private final static Singleton INSTANCE = new Singleton();

    private Singleton(){}

    public static Singleton getInstance(){
        return INSTANCE;
    }
}
懒汉式(线程不安全)

实现懒汉式单例,我们首先想到的是构造函数私有化以及获取唯一实例的方法

public class Singleton {

    private static Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            singleton = new Singleton();
        }
        return singleton;
    }
}
懒汉式(方法锁,线程安全)

于是为了保证线程安全,我们在获取实例的方法上加锁,这样的确可以保证线程安全,但是这里的同步块并非最小粒度

public class Singleton {

    private static Singleton singleton;

    private Singleton() {}

    public static synchronized Singleton getInstance() {
        if (singleton == null) {
            singleton = new Singleton();
        }
        return singleton;
    }
}
懒汉式(线程不安全)

于是我们理所当然的想到了这种错误的方式,他的问题在于若两个线程同时到达 if (singleton == null) 则会创建两个实例。

public class Singleton {

    private static Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                singleton = new Singleton();
            }
        }
        return singleton;
    }
}
懒汉式(双重检查,仍然线程不安全)

接下来的方案仍然是线程不安全的,但是理解他的问题我们需要引入指令重排序的概念。

public class Singleton2 {

    private Singleton2(){}
    
    private static Singleton2 instance;

    public static synchronized Singleton2 getInstance(){
        if(instance == null) {
            synchronized (Singleton2.class){
                if (instance == null){
                    instance = new Singleton2();
                }
            }
        }
        return instance;
    }
}
懒汉式(双重检查禁止重排序)

对于多线程下的重排序问题java给出的解决方案是volatile关键字,他保证了在写操作完成前,不会对其进行读操作。

public class Singleton2 {

    private Singleton2(){}
    
    private static volatile Singleton2 instance;

    public static synchronized Singleton2 getInstance(){
        if(instance == null) {
            synchronized (Singleton2.class){
                if (instance == null){
                    instance = new Singleton2();
                }
            }
        }
        return instance;
    }
}
静态内部类
public class Singleton2 {

    private Singleton2(){}
    
    public static class SingletonHolder{
        private static final Singleton2 INSTANCE = new Singleton2();
    }
    
    public static Singleton2 getInstance(){
        return SingletonHolder.INSTANCE;
    }
}
上一篇下一篇

猜你喜欢

热点阅读