架构之路

设计模式——单例模式

2016-12-20  本文已影响15人  SyncAny
1. 作用:

保证一个类仅有一个实例,并且提供它的全局访问点。

2. 5种常用的写法

1. 饿汉式
//最简单的单例模式
public class SingletonFactory {
    private static volatile SingletonFactory instance = null;

    private static class SingleTon {
        private static SingletonFactory instance = new SingletonFactory();
    }

    private SingletonFactory() {

    }

    public static SingletonFactory getInstance() {
        return instance;
    }

}

'优点:没有加锁,执行效率会提高。'
'缺点:类加载时就初始化,浪费内存。'
2. 懒汉式

(1)一般的懒汉式

public class SingletonFactory {
//私有的静态实例,防止被引用,设置为null,可以实现延迟加载
    private static SingletonFactory instance = null;
//私有构造方法,防止被实例化
    private SingletonFactory() {

    }
//懒汉式1:创建实例
    public static SingletonFactory getInstance() {
        if (instance == null) {
            instance = new SingletonFactory();
        }
        return instance;
    }
}

'优点:延迟加载(需要用的时候才去加载) '
'缺点:线程不安全,在多线程的时候很容易出现不同步的情况,比如在数据库对象进行频繁的读写操作。'

(2)增加同步锁:上面的方式线程不安全,可以增加同步锁

//懒汉式2:解决线程安全问题
public static synchronized SingletonFactory getInstance() {
        if (instance == null) {
            instance = new SingletonFactory();
        }
        return instance;
}

更加普遍的写法

//懒汉式2:解决线程安全问题
public static SingletonFactory getInstance() {
        if (instance == null) {
            synchronized (SingletonFactory.class) {
                instance = new SingletonFactory();
            }
        }
        return instance;
}

'优点:解决了线程不安全的问题'
'缺点:每次调用实例都需要判断同步锁,效率降低'

(3)双重检验锁(DoubleCheckLock)
有的人为解决上面效率的问题,使用了一种双重检验的方式

//双重锁定:只在第一次初始化的时候加上同步锁
public static SingletonFactory getInstance() {
        if (instance == null) {
            synchronized (SingletonFactory.class) {
                if (instance == null) {
                    instance = new SingletonFactory();
                }
            }
        }
        return instance;
}

'存在问题:
instance = new SingletonFactory();
在JVM编译的过程中会出现指令重排的优化过程,这就会导致当 instance实际上还没初始化,就可能被分配了内存空间,
也就是说会出现 instance !=null 但是又没初始化的情况,这样就会导致返回的 instance 不完整。'

'优点:在并发量不多,安全性不高的情况下或许能很完美运行单例模式'
'缺点:不同平台编译过程中可能会存在严重安全隐患'

(4)内部类的实现

//内部类实现单例模式,延迟加载,减少内存开销
public class SingletonFactory {
    private SingletonFactory(){
        
    }
    
    private static class SingleTon {
        private static SingletonFactory instance = new SingletonFactory();
    }

    public SingletonFactory getInstance() {
        return SingleTon.instance;
    }

}

'优点:延迟加载,线程安全(java中class加载时互斥的),也减少了内存消耗'

(5)枚举的方法

public enum SingletonFactory {
     /** 
     * 1.从Java1.5开始支持; 
     * 2.无偿提供序列化机制; 
     * 3.绝对防止多次实例化,即使在面对复杂的序列化或者反射攻击的时候; 
     */ 
    instance;
    
    SingletonFactory(){
        
    }
}

注意:
1. 枚举使用关键字:enum
2. 调用方式:SingletonFactory.instance.方法名

以上列举了5种单例的实现方法,下面简单的介绍一下用到的关键字:

1. volatile(非阻塞性的)
2. synchronized 关键字
  1. 获得同步锁;
  2. 清空工作内存;
  3. 从主内存拷贝对象副本到工作内存;
  4. 执行代码(计算或者输出等);
  5. 刷新主内存数据;
  6. 释放同步锁。
    所以,synchronized既保证了多线程的并发有序性,又保证了多线程的内存可见性。
上一篇 下一篇

猜你喜欢

热点阅读