刚到北京没多长时间,找工作找的太急了,从投简历到上班,第4天收到一个小公司的offer,就直接上班了,有点后悔。这个公司做android开发的就我自己,况且我也是一个菜菜菜菜鸟。算了不说了,学吧。自己有实力了在找一个好一点的公司。
今天开始学习设计模式,今天先学习单例模式。
首先来说单例模式的场景:确保某个类有且只有一个对象,避免产生多个对象来消耗资源。比如说:访问IO和数据库,以及网络访问等。
下面就直接说单例的几种模式:
1 饿汉单例模式(所谓饿汉就是特别饿了,我上来就直接吃–上来就直接new这个单例的对象)
public class EHanModel{
private static final EHanModel eHanModel= new EHanModel();
private EHanModel(){}
public static EHanModel getInstance(){
return eHanModel;
}
}
饿汉缺点:类加载时就初始化,浪费内存。
2 懒汉单例模式(特别懒,用到时才new)
public class LanHanModel{
private static LanHanModel lanHanModel;
private LanHanModel(){}
public static synchronized LanHanModel getInstance(){
if(lanHanModel == null){
lanHanModel = new LanHanModel();
}
return lanHanModel;
}
}
懒汉模式在使用时才实例化,在一定程度度上节约的资源,但是他每一次获取实例时,都会进行同步一次,虽然保证了多线程的情况下,单例的唯一性,但是会造成不必要的同步开销。
3 DoubleCheckLock
public class Singleton{
private static Singleton singleton;
private Singleton(){};
public static Singleton(){
if(singleton == null){
syschronized(Singleton.class){
if(singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
}
一般来说DCl就可以满足单例模式的需求了,但是由于Java内存模型的原因偶尔失败,在高并发的情况下,也有一定的缺陷。
4 最推崇的单例模式
public class Singleton{
private Singleton(){}
public static Singleton getInstance(){
return SingletonHolder.sInstance;
}
}
public static class SingletonHolder{
private static final Singleton sInstance = new Singleton();
}
这种方式,保证了线程安全,也保证了单例的唯一性,并且延时了单例的实例化。
5 枚举模式的单例
public enum SingletonEnum{
INSTANCE;
public void doSomething(){
system.out.println("do sth.");
}
}
枚举最大的有点就是在任何情况下都是单例的,包括反序列化。
上面几个示例如果想杜绝单例对象在反序列化时重生对象,则应该加入如下的方法:
private Object readResolve() throws ObjectStreamException{
return instance;
}
6 容器实现单例
public class Singleton{
private static Map<String,Object> map = new HashMap<String,Object>;
private Singleton(){}
public static void registerService(String string,Object object){
if(!map.containsKey(string)){
map.put(string,object);
}
}
public static Object getService(String string){
retnrn Object map.get(string);
}
}
好了平常的单例模式基本上就这些了,接下里分析一下android源码中的常用的单例模式。