JAVA设计模式Java学习笔记技术干货

单例模式

2016-12-27  本文已影响40人  Chauncey_Chen

单例模式介绍

单例模式是java创建性模式的一种,提供了一种创建对象的最佳模式.

关于指令集乱序:

在堆中的对象,还没有完全实例完成,就将地址交给了引用

设计意图:

保证一个类只有一个实例,减小内存开支,减少资源多次链接(当一个类是作为资源链接,保证这个实例为唯一实例,而不是链接一次,新建一个实例,这样能有效减少系统开销).

设计思想:

1.单例类只能有一个实例.

2.单例必须自己创建自己的唯一实例,且构造函数私有(反射除外).

3.单例必须给所有其他对象,提供这一个实例.

设计举例:

设计方式:

1.饿汉式,线程安全


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

2.懒汉式,线程不安全


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

3.懒汉式,线程安全


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

4.双重校验锁(DCL,即 double-checked locking) 线程安全

这种方式采用双锁机制,安全且在多线程情况下能保持高性能。
getInstance() 的性能对应用程序很关键。

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

}

5.静态内部类


public class InnnerClassSingleton {
    
    private InnnerClassSingleton() {
        
    };
    
    /**
     * 静态内部类,其实和一个静态外部类的内存加载方式一样.
     * 只有在将内部类被调用时,才会将SingleHolder.class文件,加载到方法区.
     * 然后将所有静态变量(区,方法)进行初始化,达到加载完成.(懒加载效果)
     * 这种方法,保证了唯一实例
     * @author ccj
     *
     */
    private static class SingleHolder{
        private final static InnnerClassSingleton INSTANCE =new InnnerClassSingleton();
        
    }
    
    public InnnerClassSingleton getInstance(){
        
        return SingleHolder.INSTANCE;
    }
}

6.枚举法

       1. 枚举中的属性必须放在最前面,一般使用大写字母表示

    2. 枚举中可以和java类一样定义方法

    3. 枚举中的构造方法必须是私有的

默认枚举实例的创建是线程安全的.(创建枚举类的单例在JVM层面也是能保证线程安全的), 所以不需要担心线程安全的问题


public enum EnumSingleton {
    
    INSTANCE;
    
    public void write(){
        
    }
    

}

总结

1.恶汉模式,当classloader加载完成.class文件后,就实例化对象,线程安全,但是没有懒加载.在没有明确要求懒加载时候,推荐使用.

2.懒汉模式,当调用时才会实例化对象,但是线程不安全.即使用sychronize同步,也会出现指令集乱序,造成多个实例产生.

3.DCL双重检验锁模式,用voliate和sychronize保证只有一个实例.但是大量的校验对比,会使效率低.

4.内部类模式,优雅的模式,懒加载,而且线程安全,所以推荐使用.

5.枚举方式,最佳的单例方式,它更简洁,自动支持序列化机制,绝对防止多次实例化。推荐使用

上一篇下一篇

猜你喜欢

热点阅读