单例模式的几种可行方式

单例的概念

  单例模式是一种对象创建模式,他用于产生一个对象的具体实例,它可以确保系统中一个类只产生一个实例。
  对于频繁使用的对象,可以省略创建对象所花费的时间,对于那些重量级的对象而言,是非常可观的一笔系统开销。由于 new 操作的次数减少,因而对系统内存的使用频率也会降低,这将减轻 GC 压力,缩短 GC 时间。
  单例模式的核心在于通过一个接口返回唯一的对象实例。单例主要需要解决的问题就是要把创建对象的权限收回来,比如说单例最重要的方法就是把类的构造方法设为 private,这就是把创建对象的权限收回来,而让类自身来完成创建自己实例的工作,然后让类提供外部可以访问这个实例的方法。
  下面是几种可选择的单例实现方式。

饿汉式:
/**
 * 饿汉式
 */
public class Singleton1 {

    private static Singleton1 instance = new Singleton1();

    private Singleton1(){

    }

    public static Singleton1 getInstance() {
        return instance;
    }

    public void method(){
        // do something.
    }

}

  饿汉模式是在类创建时就创建实例,是线程安全的。所谓饿汉,就是因为在类的实例还未被需要使用时,这个实例就被创建了,因此是不能懒加载,这是饿汉式的缺点。

懒汉DCL式:
/**
 * 懒汉DCL式
 */
public class Singleton2 {

    private static volatile Singleton2 instance;

    private Singleton2(){

    }

    public static Singleton2 getInstance(){
        if (instance == null){
            synchronized (Singleton2.class){
                if (instance == null){
                    instance = new Singleton2();
                }
            }
        }
        return instance;
    }

    public void method(){
        // do something.
    }

}

  懒汉模式,相对于饿汉模式区别,是懒汉模式的实例在类的实例需要使用时,才会被创建,是懒加载。在此之上,我们这里又用到了 DCL(double checked locking)的方式,来保证单例的线程安全,还用了 volatile 关键字来禁止 JVM 对 instance 相关的操作的指令重排序优化,保证 instance 是唯一的。

静态内部类式:
/**
 * 静态内部类式
 */
public class Singleton3 {

    private Singleton3(){

    }

    public static Singleton3 getInstance(){
        return InstanceHolder.instance;
    }

    public void method(){
        // do something.
    }

    /**
     * 持有单例的静态内部类
     */
    private static class InstanceHolder{
        static final Singleton3 instance = new Singleton3();
    }

}

  静态内部类单例是懒加载的。静态内部类实现单例主要是依靠 JVM 提供给我们的同步控制,利用 static 和 final 两个关键字, 通过 static 进行区块初始化数据,保证数据在内存中是独一份的,final 字段保证数据初始化后无法被修改。由此来保证单例的同步控制。
  静态内部类单例相对于饿汉DCL式的优点是,由于没用使用 synchronized,所以性能上会有一些优势。

枚举式:
/**
 * 枚举式
 */
public enum Singleton4 {

    INSTANCE;

    public void method(){
        // do something.
    }

}

  枚举式是利用 enum 的特性来保证其唯一性,它的优点显而易见,实现起来十分简单,同时它又是是线程安全的。

分别的使用方法:
        //饿汉式
        Singleton1.getInstance().method();
        //懒汉DCL式
        Singleton2.getInstance().method();
        //静态内部类式
        Singleton3.getInstance().method();
        //枚举式
        Singleton4.INSTANCE.method();

  以上是单例模式的几种可行方式。

猜你喜欢

转载自blog.csdn.net/weixin_42247440/article/details/81475220