美文网首页
单例模式Singleton

单例模式Singleton

作者: 丶time | 来源:发表于2018-05-08 22:13 被阅读0次

    单例模式singleton

    image

    单线程懒汉式:

    public class Singleton {  
          
        private static Singleton instance;//static意味着一个类只有一个实例!
        private Singleton() {};//私有化构造器 
        public static Singleton getInstance(){  
            if (instance==null) {  
                instance=new Singleton();//延迟创建
            }  
            return instance;  
        }  
    }
    

    延迟实例化

    单线程饿汉式:

    public class Singleton {    
          private static Singleton instance = new Singleton(); //加载类时就创建   
          private Singleton() {    
          }    
          public static Singleton getInstance() {    
              return instance;    
        }    
    

    加载类时就实例

    枚举

    public enum Singleton {      
          INSTANCE;      
          public void whateverMethod() {      
          }     
      }
    

    创建枚举实例的过程是线程安全的,所以这种写法也没有同步的问题。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。

    在需要继承的场景,它就不适用了。

    多线程懒汉

    public class Singleton    
       {    
          private volatile static Singleton singleton = null;    //volatile保证指令不被重排序!详解见下
          private Singleton()  {}    
          public static Singleton getInstance()   {    
              if (singleton== null)  {    //先检查,若有则直接返回不用加锁,提高效率
                  synchronized (Singleton.class) {    //加锁
                      if (singleton== null)  {    //如果在第一次检查到加锁之间,别的线程可能创建了实例,因此需要再检查。
                      //第一次检查实际大多是在创建了之后,避免加锁提高效率,这次检查一般在创建之初的时候。
    
    
                          singleton= new Singleton();    
                     }    
                 }    
             }    
             return singleton;    
         }    
     }
    
     singleton= new Singleton();//如果没有volatile
    

    这个new一共有三步

    1.为对象分配内存

    2.实例化对象

    3.将引用instace指向分配的内存空间

    结果:如果没有volatile,处理器重排序会导致顺序可能为1,3,2.即singleton指向了分配好的内存空间,但是对象没有实例化。

    此时,另外一个线程使用singleton,发现已经不为null了,(只是在getInstace方法加锁了,不影响其他方法)于是在另外一个方法中使用singleton,由于对象没有实例化,则会报错。

    静态内部类

     public class Singleton {    
          private Singleton() {}    
          
          //静态内部类
          private static class SingletonHolder {    
              private static final Singleton INSTANCE = new Singleton();    
          }    
          
          
          public static final Singleton getInstance() {    
              return SingletonHolder.INSTANCE;   //由静态内部类创建实例 
         }    
     }
    

    延迟加载且线程安全

    延迟加载:静态内部类在主动调用时才开始加载
    线程安全:由虚拟机保证

    相关文章

      网友评论

          本文标题:单例模式Singleton

          本文链接:https://www.haomeiwen.com/subject/eekgrftx.html