个人随笔之单例模式

为什么使用单例模式?

确保某一个类只有一个实例,避免产生多个对象消耗过多的资源, 如要访问IO和 数据库等资源

实现单例模式的几个关键点:

构造函数不对外开放,一般为private
通过一个静态方法或枚举返回单例类对象
确保单例类的对象只有一个,尤其是在多线程环境下
确保单例类对象反序列化时不会重新构建对象

饿汉模式—空间换取时间,线程安全

public class Singleton {

    private Singleton() { }

    private static Singleton instance = new Singleton();

    public static Singleton getInstance() {
        return instance;
    }
}

懒汉模式

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

在getinstance方法中加入了synchronized关键字,也就是getInstance是一个同步方法,是线程安全的,但是这种方式存在性能上的缺陷,每次调用getInstance都会进行同步,消耗不必要的资源

优点:只有在使用时才会被实例化,在一定程度上节约了资源
缺点:第一次加载时需要及时实例化,反应稍慢,最大问题是每次调用getInstance都进行同步,造成不必要的同步开销

Double check lock 模式

能够在需要时才实例化,保证线程安全,只有在第一次调用时才进行同步,以后调用getInstance都不会进行同步锁

public class Singleton {

    private static Singleton instance = null;

    private Singleton() {}

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

getInstance方法中对instance进行了两次判空,第一次主要是为了避免不必要的同步,第二层则是为了在null的情况下创建实例

instance = new Singleton();此操作并不是一个原子操作,这句代码最终会被编译成多条汇编指令

  1. 给Singleton的实例分配内存
  2. 调用Singleton的构造函数,初始化成员字段
  3. 将instance对象指向分配的内存空间(此时instance就不是null了)

由于Java编译器允许处理器乱序执行,第2,3条的指令执行顺序是无法保证的,执行顺序可能为1-2-3或1-3-2,假定A线程执行的顺序为1-3-2,当执行到第三条指令时,此时instance已经不为null,切换到B线程,判断instance不为空,直接取走了instance,使用时就会报错,double check lock模式此时就失效了。

在Java1.5以后,具体化了volatile关键字,volatile会将修改的变量值立即更新到主存中,保证变量的可见性,只需将instance定义改为private volatile static Singleton instance = null,就可以保证instance对象每次都是从主内存中读取。

优点:资源利用率高,第一次执行getInstance时单例对象才会被实例化
缺点:第一次加载时反应稍慢

静态内部类模式

public class Singleton {

    private Singleton() {}

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

第一次加载Singleton类时并不会初始化instance,只有在第一次调用getInstance方法时才会初始化instance实例,因此,第一次调用getInstance方法会触发虚拟机加载SingletonHolder类,这种方式不仅能保证线程安全,也能保证实例对象的唯一性,同时也延迟了单例的实例化,推荐用此模式

枚举单例

public enum  Singleton {
    INSTANCE;
    public void doSomething() {
        
    }
}

枚举在Java中和普通类是一样的,不仅能够拥有字段,还能有自己的方法,最重要的是默认枚举实例的创建是线程安全的,并且在任何情况下它都是一个单例。

在上述几种单例模式的实现中,在反序列化的情况下他们会出现重新穿件对象,枚举方式则不会

通过序列化可以将一个单例的实例对象写到磁盘,然后再读回来,从而有效的获得一个实例。即使构造函数是私有的,反序列化时依然可以通过特殊的途径去创建类的一个新实例。反序列化操作提供了一个很特别的钩子函数readResolve,如果要杜绝单例对象在被反序列化时重新生成对象,那么必须加入readResolve函数,也就是在readResolve函数中将单例对象返回,而不是重新生成一个新对象。

private Object readResolve() {
    return SingletonHolder.instance;
}
public class Singleton implements Serializable {

    private Singleton() {
    }

    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }

    private static class SingletonHolder {
        private static Singleton instance = new Singleton();
    }

//    private Object readResolve() {
//        return SingletonHolder.instance;
//    }


    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Singleton s = Singleton.getInstance();
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.obj"));
        oos.writeObject(s);
        oos.flush();
        oos.close();

        FileInputStream fis = new FileInputStream("Singleton.obj");
        ObjectInputStream ois = new ObjectInputStream(fis);
        Singleton s1 = (Singleton) ois.readObject();
        ois.close();
        System.out.println(s + "\n" + s1);
    }
}

//output
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa
com.miracle.thirdlibsourcecode.Singleton@6acbcfc0

不加钩子函数反序列化的时候则会创建新的对象

public class Singleton implements Serializable {

    private Singleton() {
    }

    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }

    private static class SingletonHolder {
        private static Singleton instance = new Singleton();
    }

    private Object readResolve() {
        return SingletonHolder.instance;
    }


    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Singleton s = Singleton.getInstance();
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.obj"));
        oos.writeObject(s);
        oos.flush();
        oos.close();

        FileInputStream fis = new FileInputStream("Singleton.obj");
        ObjectInputStream ois = new ObjectInputStream(fis);
        Singleton s1 = (Singleton) ois.readObject();
        ois.close();
        System.out.println(s + "\n" + s1);
    }
}

//output
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa

加了钩子函数readResolve以后,反序列化就不会重新创建对象了

而对于枚举,并不存在这个问题,因为即使反序列化也不会生成新的实例对象

通过对enum反编译发现其实enum是继承了Enum抽象类,在反序列化时,反射在通过newInstance创建对象时,会检查该类是否ENUM修饰,如果是则抛出异常,反射失败。

你可能感兴趣的:(个人随笔之单例模式)