单例模式的几种写法

一、单例模式概述

单例模式定义很简单:一个类中能创建一个实例,所以称之为单例。
那我们为什么要使用单例模式呢?

  • 那既然一个类中只能创建一个实例,那么可以说这是跟类的状态与对象无关的了。
  • 频繁创建对象、管理对象是一件耗费资源的事,我们只需要创建一个对象来用就足够了。

如果你学过J2EE,你可能知道:

  • Servlet是单例
  • Struts2是多例
  • SpringMVC是单例的

Struts2为啥设计成多例呢?

  • 这主要是由于设计层面上的问题,Struts2是基于Filter拦截类的,ognl引擎对变量是注入的,所以它要设计成多例

二、单例模式示例

编写单例模式的代码其实很简单,分为三步:

  • 将构造函数私有化
  • 在类的内部创建示例
  • 提供获取唯一实例的方法

2.1 饿汉式

根据上面的步骤,我们就可以创建单例模式了。

public class Java2y {

    // 1.将构造函数私有化,不可以通过new的方式来创建对象
    private Java2y(){}
    
    // 2.在类的内部创建实例
    private static Java2y y = new Java2y();
    
    // 3.提供获取唯一实例的方法
    public static Java2y getInstance(){
        return y;
    }
}

这种代码我们称之为“饿汉式”:

  • 一上来就创建了对象,如果该实例从始至终都没被使用过,则会造成内存浪费。

2.2 简单饿汉式

既然一上来就创建对象会造成内存浪费,那我们设计成用到的时候再创建对象

public class Java2y {

    // 1.将构造函数私有化,不可以通过new的方式来创建对象
    private Java2y(){}
    
    // 2.1先不创建对象,等用到的时候再创建
    private static Java2y y = null;
    
    // 2.2调用这个方法,创建对象
    public static Java2y getInstance(){
        // 3.如果对象为null,就创建并返回
        if(y == null){
            y = new Java2y();
        }
        return y;
    }
}

上面的代码不行吗?在单线程环境下是可行的,如果在多线程环境下就有问题了,解决方法也很简单,加锁就行。

public class Java2y {

    // 1.将构造函数私有化,不可以通过new的方式来创建对象
    private Java2y(){}
    
    // 2.1先不创建对象,等用到的时候再创建
    private static Java2y y = null;
    
    // 2.2调用这个方法,创建对象
    public static synchronized Java2y getInstance(){
        // 3.如果对象为null,就创建并返回
        if(y == null){
            y = new Java2y();
        }
        return y;
    }
}

2.3 双重检测机制(DCL)懒汉式

上面那种直接在方法上加锁的方式其实不够好,因为在方法上加了内置锁,在多线程环境下性能会比较低,所以我们可以将锁的范围缩小。

public class Java2y {
    
    private Java2y(){}
    
    private static Java2y y = null;
    
    private static Java2y getInstance(){
        if(y == null){
            // 将锁的范围缩小,提高性能 
            synchronized(Java2y.class){
                y = new Java2y();
            }
        }
        return y;
    }
}

这样写以后可行了吗?不行,因为虽然加了锁,但还是有可能创建出两个对象出来:

  • 线程1和线程2同时调用getInstance()方法,它们同时判断y==null,因为结果都为null,所以进入了if代码块。
  • 此时线程1得到CPU的控制权-->进入同步代码块-->创建对象-->返回对象
  • 线程1完成后,线程2得到了CPU控制权,一样是进入同步代码块-->创建对象-->返回对象
  • 然后很明显,最终返回了不止一个对象

然后有人又想到了:进入同步代码块时再判断一下对象是否存在就行了吧,于是有了下面的代码:

public class Java2y {
    
    private Java2y(){}
    
    private static Java2y y = null;
    
    private static Java2y getInstance(){
        if(y == null){
            // 将锁的范围缩小,提高性能 
            synchronized(Java2y.class){
                if(y == null){
                    y = new Java2y();
                }
            }
        }
        return y;
    }
}

然后这种方式又出现了重排序的问题!!!怎么解决呢?加上volatile关键字吧,volatile有内存屏障的功能!

所以说完整的DCL代码是这样子的:

public class Java2y {
    
    private Java2y(){}
    
    private static volatile Java2y y = null;
    
    private static Java2y getInstance(){
        // 这个判空是为了提高性能
        if(y == null){
            // 将锁的范围缩小,提高性能 
            synchronized(Java2y.class){
                if(y == null){
                    y = new Java2y();
                }
            }
        }
        return y;
    }
}

2.4 静态内部类懒汉式

它的原理是这样的:

  • 当任何一个线程第一次调用getInstance()时,都会使SingletonHolder被加载和被初始化,此时静态初始化器将执行Singleton的初始化操作。(被调用时才进行初始化!)
  • 初始化静态数据时,Java提供了的线程安全性保证。(所以不需要任何的同步)
public class Java2y {
    
    private Java2y (){}
    
    // 使用私有静态内部类实现懒加载
    private static class LazyHolder {
        private static final Java2y INSTANCE = new Java2y();
    }
    
    public static Java2y getInstance(){
        return LazyHolder.INSTANCE;
    }
}

这种方式非常推荐使用!!!

2.5 枚举方式

public enum Java2y {
    JAVA_2_y,
}

这种实现:

  • 简单
  • 防止多次实例化,即使是在复杂序列化或者反射攻击时也很安全
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容