设计模式:第五篇--单例模式

单例模式:确保一个类只有一个实例,并提供要给全局访问点。
适用场景:共享的资源,比如数据库连接池,缓存等。我们可以尝试将需要释放资源的对象用单例模式管理起来,就想连接池和线程池那样。

  • 延迟实例化+线程不安全
  • 非延迟实例化+线程安全
  • 延迟实例化+线程安全
  • 总结
延迟实例化+线程不安全

经典的单例模式采取了“延迟实例化”的方式。即当程序用到的时候才会创建他,没有用到就不创建了。

/**
 * Project 
 * Created by jorgezhong on 2018/9/25 9:42.
 * 

* 经典单例模式 * - 利用静态成员变量记录单例对象 * - 私有构造方法 * - 提供静态方法获取成员变量 */ public class Singleton { private static Singleton uniqueInstance; private Singleton() { } public static Singleton getInstance() { //延迟实例化 if (uniqueInstance == null) { uniqueInstance = new Singleton(); } return uniqueInstance; } }

非延迟实例化+线程安全

问题:出现多线程问题,多线程情况下,可能会实例化多个对象。
思考:可使用同步锁synchronized,这样会消耗性能,其实使用的时候,只有第一次实例化的时候才需要保持同步,之后便不需要了,因此如果加synchronized的话,每次调用方法都要加所,会消耗性能。那既然这样的话,不使用延迟的方式不久没有线程安全问题了吗?静态初始化的时候给成员变量赋值就好了。

/**
 * Project 
 * Created by jorgezhong on 2018/9/25 10:28.
 * 

* 静态初始化的时候:初始化单例对象,解决线程安全问题 */ public class Singleton { private static Singleton uniqueSingleton = new Singleton(); private Singleton() { } public static Singleton getInstacne() { return uniqueSingleton; } }

延迟实例化+线程安全

问题:那如果我非得要使用延迟实例化的方式呢?有时候由于实例化对象太大,而且程序运行中很多时候并不会用到。所以想要使用延迟实例化的方式。
思考:那么还得使用synchronized的方式呀。不过,需要加上双重判断,这样只需要在实例化完成之前执行锁代码就可以了,如果在成员变量中加上volatile修饰就更好了。

/**
 * Project 
 * Created by jorgezhong on 2018/9/25 10:41.
 *
 * 双重检查
 */
public class Singleton {

    //volatile:确保可见性,保证每个线程拿到的值是最新的
    private volatile static Singleton uniqueSingleton;

    private Singleton() {
    }

    public static Singleton getInstance() {

        //第一次检查,进入锁代码
        if (uniqueSingleton == null) {
            synchronized (Singleton.class) {
                //第二次锁内检查,确保线程安全
                if (uniqueSingleton == null) {
                    uniqueSingleton = new Singleton();
                }
            }
        }

        return uniqueSingleton;

    }

}
总结:

单例模式比较简单,使用场景也时比较常见的,有时候我们会引入一些工具像nosql这些,经常需要去获取连接,管理这些连接的对象可以时使用单例的,这样我们就不需要为资源未释放而烦恼。缓存控制和日志对象也是可以使用的。

你可能感兴趣的:(设计模式:第五篇--单例模式)