JUC知识点总结(四)五种单例模式的写法

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 效果时,使用线程安全的懒汉式。如果涉及到反序列化创建对象时,可以尝试使用枚举方式。如果有其他特殊的需求,可以考虑使用双检锁方式。

下一篇
JUC知识点总结(五)线程的生命周期

发布了54 篇原创文章 · 获赞 11 · 访问量 2万+

猜你喜欢

转载自blog.csdn.net/dong_W_/article/details/105013264