9. 五种单例模式的写法
饿汉式:
public class Singleton{
private static Singleton instance = new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return instance;
}
}
懒汉式(Synchronized):
public class Singleton{
private static Singleton instance;
private Singleton(){}
public static synchronized Singleton getInstance(){
if(instance==null){
instance=new Singleton();
}
return instance;
}
}
懒汉式(双重检查锁DCL,即 double-checked locking)
public class Singleton{
private static volatile Singleton instance;
private Singleton(){}
public static Singleton getInstance(){
if(instance==null){
synchronized(Singleton.class){
if(instance==null){
instance = new Singleton();
}
}
}
return instance;
}
}
在单例的懒汉模式中,必须给实例添加volatile修饰符
原因:在构造实例时,对象引用指针的操作和初始化操作可能会被重排序, 这就导致在if(instance==null)的时候认为对象已经创建,但这个时候还没有进行初始化
1.分配对象的内存空间2.初始化对象3.设置instance指向内存空间4.初次访问对象
3和2可能会被重排序,导致1342这样的问题。
解决办法:①禁止重排序(volitale)②允许重排序但非构造线程不可见(static class)
懒汉式(静态内部类)
public class Singleton{
private Singleton(){}
public static Singleton getInstance(){
return SingletonHolder.instance;
}
private static class SingletonHolder{
private static Singleton instance = new Singleton();
}
}
虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕。 (复习JVM类的初始化时机)
懒汉式(枚举)
public enum Singleton {
INSTANCE;
public void doingSomething() {
}
}
优点:不仅能避免多线程同步问题,而且还自动支持序列化机制,防止反序列化重新创建新的对象,绝对防止多次实例化
缺点: 不能通过反射来调用私有构造方法
总结: 饿汉方式绝对线程安全。明确要求实现 lazy loading 效果时,使用线程安全的懒汉式。如果涉及到反序列化创建对象时,可以尝试使用枚举方式。如果有其他特殊的需求,可以考虑使用双检锁方式。