你真的会写单例模式吗?

前言

单例模式是 Java 设计模式中最简单的一种,只需要一个类就能实现单例模式,但是,你可不能小看单例模式,虽然从设计上来说它比较简单,但是在实现当中你会遇到非常多的坑,所以,系好安全带,上车。

什么是单例模式

单例模式(Singleton)就是一个类在程序运行中只实例化一次,创建一个全局唯一对象,有点像 Java 的静态变量,但是单例模式要优于静态变量,静态变量在程序启动的时候JVM就会进行加载,如果不使用,会造成大量的资源浪费。单例模式能够实现懒加载,能够在使用实例的时候才去创建实例。开发工具类库中的很多工具类都应用了单例模式,比如线程池、缓存、日志对象等,它们都只需要创建一个对象。

在开发中,会经常遇到一个全局使用的类频繁地创建与销毁,这会非常浪费系统的内存资源,而且容易导致错误甚至一定会产生错误,所以我们单例模式所期待的目标或者说使用它的目的,是为了尽可能的节约内存空间,减少无谓的GC消耗,并且使应用可以正常运作。

如何实现单例模式

  • 静态并私有化实例对象
  • 提供一个公共的静态方法访问私有的静态实例,用来返回唯一实例对象
  • 私有化构造方法,禁止通过构造方法创建实例

单例有什么好处

  • 只有一个对象,内存开支少、性能好
  • 避免对资源的多重占用
  • 在系统设置全局访问点,优化和共享资源访问

以下介绍常见的几种单例模式实现方式

单例模式的写法有饿汉模式、懒汉模式、双重检查锁模式、静态内部类单例模式、枚举类实现单例模式五种方式,其中懒汉模式、双重检查锁模式,如果你写法不当,在多线程情况下会存在不是单例或者单例出异常等问题,具体的原因,在后面的对应处会进行说明。我们从最基本的饿汉模式开始我们的单例编写之路。

饿汉模式

所谓饿汉,自然是非常迫切获取食物,这里的食物所指的自然就是我们的单例对象了。饿汉模式是已一种简单粗暴的方式创建单例对象,在定义静态属性时,直接实例化了对象。

public class SingletonObjectHunger {
    /**
     * 私有静态实例
     */
    private static final SingletonObjectHunger INSTANCE = new SingletonObjectHunger();

    /**
     * 私有化构造函数
     */
    private SingletonObjectHunger(){
        // code
    }

    /**
     * 提供公开获取实例接口
     * @return 单例对象
     */
    public static SingletonObjectHunger getInstance(){
        return INSTANCE;
    }
}

优点

  • 由于使用了static关键字,保证了在引用这个变量时,关于这个变量的所以写入操作都完成,所以保证了JVM层面的线程安全

缺点

  • 没有实现懒加载的效果,如果一个类比较大,我们在初始化的时就加载了这个类,但是如果我们没有使用这个类,这就导致了内存空间的浪费。

懒汉模式

所谓懒汉,就是以偷懒的方式创建单例。这种方式在类初始化的时候不会创建实例,只有在获取使用实例的时候才会创建实例,这样就解决了饿汉模式的空间浪费问题,但是也引入了其他问题。下面介绍一种懒汉模式的写法

public class SingletonObjectLazy {

    /**
     * 私有静态实例 未初始化实例
     */
    private static SingletonObjectLazy instance;

    /**
     * 私有化构造函数
     */
    private SingletonObjectLazy(){
        // code
    }

    /**
     * 提供公开获取实例接口
     * @return 单例对象
     */
    public static SingletonObjectLazy getInstance(){
        // 先判断实例是否为空,如果实例为空,则实例化对象
        if (instance == null) {
            instance = new SingletonObjectLazy();
        }
        return instance;
    }
}

上面是一种懒汉模式的实现方式,但在多线程情况是线程不安全的。这种写法保证不了单列模式,可能出现多实例的情况。下面分析出现多实例的原因

       1 if (instance == null) {
       2     instance = new SingletonObjectLazy();

问题就出在上面这个代码片段内。假设有两个线程同时进入到 1 这个位置,因为没有任何资源保护措施,所以两个线程可以同时判断的instance都为空,都将去执行 2 的实例化代码,这样就实例化了两份实例,所以会出现多份实例的情况。

通过上面的分析我们已经知道出现多份实例的原因,如果我们在创建实例的时候进行资源保护,不就可以解决多份实例的问题了吗?确实如此,我们给getInstance()方法加上synchronized关键字,使得getInstance()方法成为同步方法,能够解决多份实例的问题。加上synchronized关键字之后代码如下:

    public synchronized static SingletonObjectLazy getInstance(){

这样,在某一时刻永远只会有一个线程能执行这个方法,初始化完毕后,其他线程直接获取这个实例。这样似乎解决了线程安全问题。但是同样引入了一个新的问题,加锁之后会使得程序变成串行化,只有抢到锁的线程才能去执行这段代码块,这会使得系统的并发性能大大下降。

优点

  • 避免了饿汉模式的缺点,实现了懒加载,节约了内存空间

缺点

  • 在不加锁的情况下,线程不安全,可能出现多份实例
  • 在加锁的情况下,使程序串行化,导致系统存在严重并发性能问题

双重检查锁模式

再来讨论一下懒汉模式中加锁的问题,对于getInstance()方法来说,绝大部分的操作都是读操作,读操作是线程安全的,所以我们没必让每个线程必须持有锁才能调用该方法,我们需要去除这个同步锁。由此也产生了一种新的实现模式:双重检查锁模式。这种模式完美解决了上面的问题。

public class SingletonObjectCheck {
    /**
     * 定义静态变量时,未初始化实例
     * 要解决双重检查锁模式带来空指针异常的问题,只需要使用volatile关键字,volatile关键字严格遵循happens-before原则,即在读操作前,写操作必须全部完成。
     */
    private volatile static SingletonObjectCheck instance;

    /**
     * 私有化构造函数
     */
    private SingletonObjectCheck(){

    }

    public static SingletonObjectCheck getInstance(){
        // 使用时,先判断实例是否为空,如果实例为空,则实例化对象
        if (instance == null) {
            // 使用互斥锁 只允许单一线程初始化实例
            synchronized (SingletonObjectCheck.class){
                if (instance == null) {
                    instance = new SingletonObjectCheck();
                }
            }
        }
        return instance;
    }
}

这种写法在去除了方法上的同步锁,采用互斥锁去创建实例。假设同时有两个线程通过了第一次检查,进入到了互斥锁,线程A获取到这个锁,线程B则阻塞等待。线程A执行初始化对象后,释放锁。线程B获得锁,由于有二次检查,实例已初始化就不会去再次初始化对象。

扩展:上面的双重检测锁模式看上去完美无缺,其实是存在问题,在多线程的情况下,可能会出现空指针问题,出现问题的原因是JVM在实例化对象的时候会进行优化和指令重排操作。如果构造函数中操作比较多时,为了提升效率,JVM 会在构造函数里面的属性未全部完成实例化时,就返回对象。双重检测锁出现空指针问题的原因就是出现在这里,当某个线程获取锁进行实例化时,其他线程就直接获取实例使用,由于JVM指令重排序的原因,其他线程获取的对象也许不是一个完整的对象,所以在使用实例的时候就会出现空指针异常问题。

关于指令重排,感兴趣的同学可以自行了解。

添加volatile关键字之后的双重检查锁模式就比较完美了,能够保证在多线程的情况下线程安全也不会有性能问题。

静态内部类单例模式

静态内部类不依赖外部类,是一种很特殊的内部类。在创建静态内部类的时候,不需要外部类对象的引用。你可以把它当做*类。

静态内部类单例模式也称单例持有者模式,实例由内部类创建,由于 JVM 在加载外部类的过程中, 是不会加载静态内部类的, 只有静态内部类的属性/方法被调用时才会被加载, 并初始化其静态属性。静态属性由static修饰,保证只被实例化一次,并且严格保证实例化顺序。

public class SingletonObjectStaticInner {

    /**
     * 私有化构造方法
     */
    private SingletonObjectStaticInner() {
    }

    /**
     * 单列持有者
     */
    private static class InstanceHolder{
        private static final SingletonObjectStaticInner INSTANCE = new SingletonObjectStaticInner();
    }

    public static SingletonObjectStaticInner getInstance(){
        return InstanceHolder.INSTANCE;
    }
}

这里静态内部类为私有,只有这个外部类能访问。这种方法不仅能确保线程安全,也能保证单例的唯一性,同时也延迟了单例的实例化。

枚举类实现单例模式

枚举类实现单例模式是 effective java 作者极力推荐的单例实现模式,因为枚举类型是线程安全的,并且只会装载一次,设计者充分的利用了枚举的这个特性来实现单例模式,枚举的写法非常简单,而且枚举类型是所用单例实现中唯一一种不会被破坏的单例实现模式。

public class SingletonObjectEnum {

    /**
     * 私有化构造函数
     */
    private SingletonObjectEnum() {
    }

    /**
     * 定义一个静态枚举类
     */
    static enum SingletonEnum {
        /**
         * 创建一个枚举对象,该对象天生为单例
         */
        INSTANCE;
        private SingletonObjectEnum SingletonObjectEnum;

        /**
         * 私有化枚举的构造函数
         */
        private SingletonEnum() {
            SingletonObjectEnum = new SingletonObjectEnum();
        }

        public SingletonObjectEnum getInstance() {
            return SingletonObjectEnum;
        }
    }

    public static SingletonObjectEnum getInstance() {
        return SingletonEnum.INSTANCE.getInstance();
    }
}

这里的静态枚举内部类与之前的静态内部类都是静态内部类,不依赖外部类。枚举值天生为单例,保证了实例的唯一性,私有化枚举构造函数,阻止再次实例化。

扩展:破坏单例模式的方法及解决办法

  • 除枚举方式外, 其他方法都会通过反射的方式破坏单例,反射是通过调用构造方法生成新的对象,所以如果我们想要阻止单例破坏,可以在构造方法中进行判断,若已有实例, 则阻止生成新的实例,解决办法如下:
private SingletonObject(){
    if (instance !=null){
        throw new RuntimeException("实例已经存在,请勿重复初始化");
    }
}
  • 如果单例类实现了序列化接口Serializable, 就可以通过反序列化破坏单例,所以我们可以不实现序列化接口,如果非得实现序列化接口,可以重写反序列化方法readResolve(), 反序列化时直接返回相关单例对象。
public Object readResolve() throws ObjectStreamException {
        return instance;
    }
上一篇:CentOS 6.5上安装MariaDB


下一篇:【算法与数据结构】查找二叉树的实现