实现Singleton模式

单例模式的两种实现方式

1、双端检测

  • 通过volatile来保证线程之间变量的可见性
  • 用两个if判断,第一个判断来减少synchronized导致的低效率
//双重检查
/**
 * Double-Check双重检查,判断if(singleton==null)
 * 避免反复进行方法同步
 * 线程安全;延迟加载;效率较高
 *
 */
public class Singleton05 {

    public static void main(String[] args) {
      Singleton instance= Singleton.getInstance();
       Singleton instance1= Singleton.getInstance();
        System.out.println(instance==instance1);//true

        System.out.println(instance.hashCode()==instance1.hashCode());//true
    }
}

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;
    }

}

2、通过类装载的机制

看注释:

/**
 *1.这种方式采用了类装载的机制来保证初始化实例时只有一个线程
 *2.静态内部类方式在Singleton类被装载时并不会立即实例化,而是在需要实例化时,调用getInstance方法,
 * 才会装载SingletonInstance类,从而完成Singleton的实例化
 *3.类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的
 *4.避免线程不安全,利用静态类内部类特点实现延迟加载,效率高
 */
public class Singleton06 {

    public static void main(String[] args) {
        Singleton instance=Singleton.getInstance();
       Singleton instance1=Singleton.getInstance();
        System.out.println(instance==instance1);//true

        System.out.println(instance.hashCode()==instance1.hashCode());//true
    }
}
class Singleton{
    private static volatile  Singleton instance;

    private Singleton(){}

    //写一个静态内部类,该类中有一个静态属性Singleton
    private static class SingletonInstance{
        private static final Singleton INSTANCE=new Singleton();
    }

    //提供一个静态的公有方法,直接返回SingletonInstance.INSTANCE
    public static synchronized Singleton getInstance(){
        return SingletonInstance.INSTANCE;
    }
}

 

posted @ 2019-09-21 22:49  _SpringCloud  阅读(5)  评论(0编辑  收藏  举报  来源