设计模式笔记二单例模式

2017-03-10  本文已影响9人  summer_lz

每日一文:

捭之者,料其情也;阖之者,结其诚也。
开放的话术,观察其真性情;封闭的话术,坚定其内心。

单例模式

设计原则:

一般一个类能否做成单例,最容易区别的地方就在于,这些类,在应用中如果有两个或者两个以上的实例会引起错误,又或者我换句话说,就是这些类,在整个应用中,同一时刻,有且只能有一种状态。
一般实践当中,有很多应用级别的资源会被做成单例,比如配置文件信息,逻辑上来讲,整个应用有且只能在同在时间有一个,当然如果你有多个,这可能并不会引起程序级别错误,这里指的错误特指异常或者ERROR。但是当我们试图改变配置文件的时候,问题就出来了。

public class MySingleton {
//这里的volatile很关键
private volatile static MySingleton singleton;

public static MySingleton getMySingleton(){
    //第一次检查
    if (singleton == null) {
        synchronized (MySingleton.class) {
            if (singleton == null) {
                singleton = new MySingleton();
            }
        }
    }
    return singleton;
 }
}

这次看起来既解决了线程安全问题,又不至于每次调用getInstance()都会加锁导致降低性能。看起来是一个完美的解决方案,事实上是这样的吗?
很遗憾,事实并非我们想的那么完美。Java平台内存模型中有一个叫“无序写”(out-of-order writes)的机制。正是这个机制导致了双重检查加锁方法的失效。这个问题的关键在上面代码上的第5行:instance = new Singleton();这行其实做了两个事情:
1.调用构造方法,创建了一个实例
2.把这个实例赋值给instance这个实例变量。可问题就是,这两步jvm是不保证顺序的。也就是说。可能在调用构造方法之前,instance已经被设置为非空了。下面我们一起来分析一下:

父子:静态属性-静态代码块-非静态属性-构造方法,这是总体加载顺序,其中静态属性静态代码块在Static声明的时候开始调用且只加载一次,而静态内部类只有在通过getInstance()方法调用时才会加载而且也只加载一次!而这就是下面静态内部类实现线程安全的单例模式的理论基础。

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

扩展Volatile

上一篇 下一篇

猜你喜欢

热点阅读