设计模式之单例模式

 
 

一.单例模式的介绍

单例模式是应用最广的模式之一,在应用这个模式的时候,单例对象的类必须保证只有一个实例存在。在android中的应用场景例如整个app只有一个application对象,只有一个ImageLoader对象等。

二.单例模式下的各种实现方式

1.饿汉模式

public class Singleton {  
   private Singleton() {}  //在该类初始化的时候就会自行实例化                           
   private static final Singleton single = new Singleton();   
   public static Singleton getInstance() {  
      return single;  
   } 
}

2.懒汉模式

public class Singleton { 
    private Singleton(){} 
    private final static Singleton mInstance; //没有care线程安全的问题 
    public static Singleton getInstance() { 
        if(mInstance == null){ 
            mInstance = new Singleton(); 
        } 
        return mInstance; 
    }
}

Tips:不论是饿汉模式还是懒汉模式,可能在你的app中都占有一席之地。那么我们先来对比下两者的区别,首先饿汉模式会在该类初始化的时候就自动实例化,而懒汉模式则会在对应调用getInstance方法时才会对应的实例化,实现了实例的延时加载。设想如果该实例在app中不一定被使用到,那么使用懒汉模式就可以节省内存。但是懒汉模式会在第一次获取实例时较为耗时,饿汉模式由于在初始化类时就进行了实例化,第一次获取实例就不会耗时。

以上是针对饿汉模式和懒汉模式之间的区别做的分析,接下来我们来关注之前代码中对于懒汉模式线程不安全的问题。分别提供以下几种解决方案来进行对比:

2.1.在getInstance方法上加同步锁
public class Singleton { 
     private Singleton(){} 
     private final static Singleton mInstance; //加上同步锁 
     public static synchronized Singleton getInstance() { 
        if(mInstance == null){ 
             mInstance = new Singleton(); 
        } 
        return mInstance; 
     }
}

这种方法虽然解决了线程安全的问题,但是单例模式一般都是应用在一些会被频繁调用的场景上的,如果在每次获取实例的时候都需要去进行线程同步,那会增加不小的开销,会使单例的获取变的缓慢,这样就得不偿失了。那么我们继续改进,看下面的方法:

2.2.Double Check Lock(DCL)实现单例
public class Singleton { 
     private Singleton(){} 
     private final static Singleton mInstance; 
     /*双重锁定:只在第一次初始化的时候加上同步锁*/  
     public static Singleton getInstance() { 
          if(mInstance == null){ 
              synchronized(Singleton.class){ 
                  if(mInstance == null){ 
                        mInstance = new Singleton(); 
                  }
              }
         } 
         return mInstance; 
     }
}

这种双重锁定的方式,避免了每次获取实例时不必要的同步操作,只在第一次获取实例的时候才进行同步,将开销减到了最小,并且保证了线程安全。但是,真的是线程安全了么?问题其实出在mInstance = new Singleton();这句代码,虽然它只是一句代码,但是实际上它不是一个原子操作,这句代码最终会被编译成多条汇编指令,它大致做了3件事情:
(1)给Singleton的实例分配内存;
(2)调用Singleton()的构造函数,初始化成员字段;
(3)将mInstance对象指向分配的内存空间(此时mInstance就不是null了)。
由于Java编译器允许处理器乱序执行,以及JDK1.5之前JMM(Java Memory Model,即Java内存模型)中Cache、寄存器到内存回写顺序的规定,上面的第二和第三的顺序是无法保证的。也就是说,执行顺序可能是1-2-3也可能是1-3-2。如果是后者,并且在3执行完毕、2未执行之前,被切换到另一个线程上,就会出问题。但是在你的app没有太多的高并发存在时,这种模式已经可以完全满足大多数开发者的需求。那么一定还有更好的:

2.3.静态内部类单例模式
public class Singleton { 
     private Singleton(){} 
     private final static Singleton mInstance; 
     public static Singleton getInstance() { 
         return SingletonHolder.mInstance; 
     } 
     private static class SingletonHolder { 
         private final static Singleton mInstance = new Singleton(); 
     }
}

第一次加载Singleton类的时候并不会初始化mInstance,只有在第一次调用getInstance方法时才会导致mInstance被初始化。因此,第一次调用getInstance方法会导致虚拟机加载SingletonHolder类,这种方式不仅能够确保线程安全,也能够保证单例对象的唯一性,同时也延迟了单例的实例化,所以这是推荐使用的单例模式实现方式。Tip:java中的枚举其实也是单例的一种实现方式



作者:tcsp
链接:http://www.jianshu.com/p/bd91d613603a
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值