几种线程安全单例实现的分析
1.介绍
单例模式有很多实现方法,饿汉、懒汉、静态内部类、枚举类,试分析每种实现下获取单例对象(即调用getInstance)时的线程安全,并思考注释中的问题。
- 饿汉式:类加载就会导致该单实例对象被创建。
- 懒汉式:类加载不会导致该单实例对象被创建,而是首次使用该对象时才会创建。
2.分析几种不同的实现,判断其线程安全性
实现1
问题1:阻止子类继承该类,破坏了单例方法。
问题2:创建对象可以不用new实现,采用反序列化创建对象,但反序列化会破环单例。
要添加readResolve()方法,返回类中的实例,而不是字节码实例。
public Object readResolve() {
return INSTANCE;
}
问题3:如果设置成public,那么外部可以调用该方法创建对象。不能防止通过暴力反射的方法创建新实例。
问题4:类加载阶段对成员变量的赋值都是安全的。
问题5:可以封装,提供懒惰的初始化。
实现2
问题1:
枚举类的静态成员变量,单实例的。
问题2:
静态成员变量在类加载阶段完成,不存在线程安全问题。
问题3:
不能用反射破坏单例。
问题4:
枚举类默认实现序列化接口,可以避免反序列化破坏单例。
问题5:
枚举单例属于饿汉式。
问题6:
加构造方法。
实现3
这里是线程安全,但是每次调用都会加锁,效率低。
实现4
问题1:
加上volatile可以防止指令重排到的空的实例。
问题2:
首次判断没有加synchronized,可以提高代码的效率。
问题3:
防止阻塞中的线程创建新的实例,加上判断之后,阻塞中的线程发现其他已经创建好实例时,会结束。
实现5
问题1:
属于饿汉式,使用时调用方法创建。
问题2:
没有并发问题,创建时jvm保证安全性。