Java --- 单例模式(二) 七种写法

来源:互联网 发布:淘宝号如何实名认证 编辑:程序博客网 时间:2024/06/10 19:05

第一种(懒汉,线程不安全):

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

这种写法lazy loading很明显,但是致命的是在多线程不能正常工作。

第二种(懒汉,线程安全):

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

这种写法能够在多线程中很好的工作,而且看起来它也具备很好的lazy loading,但是,遗憾的是,效率很低,99%情况下不需要同步。

第三种(饿汉):

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

这种方式基于classloder机制避免了多线程的同步问题,不过,instance在类装载时就实例化,虽然导致类装载的原因有很多种,在单例模式中大多数都是调用getInstance方法, 但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化instance显然没有达到lazy loading的效果。

第四种(饿汉,变种):

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

表面上看起来差别挺大,其实更第三种方式差不多,都是在类初始化即实例化instance。

第五种(静态内部类):

public class Singleton {    private static class SingletonHolder {        private static final Singleton INSTANCE = new Singleton();    }    private Singleton() {    }    public static final Singleton getInstance() {        return SingletonHolder.INSTANCE;    }} 

这种方式同样利用了classloder的机制来保证初始化instance时只有一个线程,它跟第三种和第四种方式不同的是(很细微的差别):第三种和第四种方式是只要Singleton类被装载了,那么instance就会被实例化(没有达到lazy loading效果),而这种方式是Singleton类被装载了,instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonHolder类,从而实例化instance。想象一下,如果实例化instance很消耗资源,我想让他延迟加载,另外一方面,我不希望在Singleton类加载时就实例化,因为我不能确保Singleton类还可能在其他的地方被主动使用从而被加载,那么这个时候实例化instance显然是不合适的。这个时候,这种方式相比第三和第四种方式就显得很合理。

第六种(枚举):

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

这种方式是Effective Java作者Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象,可谓是很坚强的壁垒啊,不过,个人认为由于1.5中才加入enum特性,用这种方式写不免让人感觉生疏,在实际工作中,我也很少看见有人这么写过。

第七种(双重校验锁):

public class Singleton {    private volatile static Singleton singleton;    private Singleton() {    }    public static Singleton getSingleton() {        if (singleton == null) {            synchronized (Singleton.class) {                if (singleton == null) {                    singleton = new Singleton();                }            }        }        return singleton;    }}  

这个是第二种方式的升级版,俗称双重检查锁定

总结

有两个问题需要注意:
1、如果单例由不同的类装载器装入,那便有可能存在多个单例类的实例。假定不是远端存取,例如一些servlet容器对每个servlet使用完全不同的类 装载器,这样的话如果有两个servlet访问一个单例类,它们就都会有各自的实例。
2、如果Singleton实现了java.io.Serializable接口,那么这个类的实例就可能被序列化和复原。不管怎样,如果你序列化一个单例类的对象,接下来复原多个那个对象,那你就会有多个单例类的实例。

对第一个问题修复的办法是:

private static Class getClass(String classname) throws ClassNotFoundException {        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();        if (classLoader == null) {            classLoader = Singleton.class.getClassLoader();        }        return (classLoader.loadClass(classname));}

对第二个问题修复的办法是:

public class Singleton implements java.io.Serializable {    public static Singleton INSTANCE = new Singleton();    protected Singleton() {    }    private Object readResolve() {        return INSTANCE;    }}   

对我来说,我比较喜欢第三种和第五种方式,简单易懂,而且在JVM层实现了线程安全(如果不是多个类加载器环境),一般的情况下,我会使用第三种方式,只有在要明确实现lazy loading效果时才会使用第五种方式,另外,如果涉及到反序列化创建对象时我会试着使用枚举的方式来实现单例,不过,我一直会保证我的程序是线程安全的,而且我永远不会使用第一种和第二种方式,如果有其他特殊的需求,我可能会使用第七种方式,毕竟,JDK1.5已经没有双重检查锁定的问题了。

========================================================
不过一般来说,第一种不算单例,第四种和第三种就是一种,如果算的话,第五种也可以分开写了。所以说,一般单例都是五种写法。懒汉,恶汉,双重校验锁,枚举和静态内部类。

原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 绿地球蔫了怎么办 煤气灶火太大了怎么办 朗诵背景音乐不够长怎么办 孩子不听话家长应该怎么办 总是对生活无欲无求怎么办 三角梅花不够艳怎么办 写不出说课中的教学过程怎么办 ps保存类型错误怎么办 word忘点保存怎么办 上台讲ppt紧张怎么办 ppt用wps打不开怎么办 拉屎拉绿色的怎么办 初中课外知识弱怎么办 孩子做作业不会怎么办 孩子有阅读障碍怎么办 英语一点都不会怎么办? 孩子抄作文交怎么办 语文阅读题差怎么办 怎么办婚介所营业执照 说话速度太快怎么办 初二期末退步很大怎么办 word文档没了怎么办 荷叶的叶子枯萎怎么办 被领导冤枉了怎么办 考狱警视力不够怎么办 幼儿园小朋友说脏话怎么办 初中成绩差高中怎么办 初中孩子不学习怎么办 跟大学班长不好怎么办 初中毕业考不上高中怎么办 客人要求加床怎么办 发现浓烟或明火怎么办 被误会谈恋爱要怎么办 小学作文不会写怎么办 高考档案丢了怎么办 考生档案丢了怎么办 对待不认识的人怎么办 妈妈头发白了怎么办 身无分文欠20万怎么办 心事太重睡不着怎么办 苹果手机存储空间不足怎么办