设计模式——单例模式

1. 作用:

保证一个类仅有一个实例,并且提供它的全局访问点。

2. 5种常用的写法

1. 饿汉式
//最简单的单例模式
public class SingletonFactory {
    private static volatile SingletonFactory instance = null;

    private static class SingleTon {
        private static SingletonFactory instance = new SingletonFactory();
    }

    private SingletonFactory() {

    }

    public static SingletonFactory getInstance() {
        return instance;
    }

}

'优点:没有加锁,执行效率会提高。'
'缺点:类加载时就初始化,浪费内存。'
2. 懒汉式

(1)一般的懒汉式

public class SingletonFactory {
//私有的静态实例,防止被引用,设置为null,可以实现延迟加载
    private static SingletonFactory instance = null;
//私有构造方法,防止被实例化
    private SingletonFactory() {

    }
//懒汉式1:创建实例
    public static SingletonFactory getInstance() {
        if (instance == null) {
            instance = new SingletonFactory();
        }
        return instance;
    }
}

'优点:延迟加载(需要用的时候才去加载) '
'缺点:线程不安全,在多线程的时候很容易出现不同步的情况,比如在数据库对象进行频繁的读写操作。'

(2)增加同步锁:上面的方式线程不安全,可以增加同步锁

//懒汉式2:解决线程安全问题
public static synchronized SingletonFactory getInstance() {
        if (instance == null) {
            instance = new SingletonFactory();
        }
        return instance;
}

更加普遍的写法

//懒汉式2:解决线程安全问题
public static SingletonFactory getInstance() {
        if (instance == null) {
            synchronized (SingletonFactory.class) {
                instance = new SingletonFactory();
            }
        }
        return instance;
}

'优点:解决了线程不安全的问题'
'缺点:每次调用实例都需要判断同步锁,效率降低'

(3)双重检验锁(DoubleCheckLock)
有的人为解决上面效率的问题,使用了一种双重检验的方式

//双重锁定:只在第一次初始化的时候加上同步锁
public static SingletonFactory getInstance() {
        if (instance == null) {
            synchronized (SingletonFactory.class) {
                if (instance == null) {
                    instance = new SingletonFactory();
                }
            }
        }
        return instance;
}

'存在问题:
instance = new SingletonFactory();
在JVM编译的过程中会出现指令重排的优化过程,这就会导致当 instance实际上还没初始化,就可能被分配了内存空间,
也就是说会出现 instance !=null 但是又没初始化的情况,这样就会导致返回的 instance 不完整。'

'优点:在并发量不多,安全性不高的情况下或许能很完美运行单例模式'
'缺点:不同平台编译过程中可能会存在严重安全隐患'

(4)内部类的实现

//内部类实现单例模式,延迟加载,减少内存开销
public class SingletonFactory {
    private SingletonFactory(){
        
    }
    
    private static class SingleTon {
        private static SingletonFactory instance = new SingletonFactory();
    }

    public SingletonFactory getInstance() {
        return SingleTon.instance;
    }

}

'优点:延迟加载,线程安全(java中class加载时互斥的),也减少了内存消耗'

(5)枚举的方法

public enum SingletonFactory {
     /** 
     * 1.从Java1.5开始支持; 
     * 2.无偿提供序列化机制; 
     * 3.绝对防止多次实例化,即使在面对复杂的序列化或者反射攻击的时候; 
     */ 
    instance;
    
    SingletonFactory(){
        
    }
}

注意:
1. 枚举使用关键字:enum
2. 调用方式:SingletonFactory.instance.方法名

以上列举了5种单例的实现方法,下面简单的介绍一下用到的关键字:

1. volatile(非阻塞性的)
  • volatile的特性:当我们声明共享变量为volatile后,对这个变量的读/写将会很特别。
  • 理解volatile特性的一个好方法是:把对volatile变量的单个读/写,看成是使用同一个监视器锁对这些单个读/写操作做了同步。
  • 它的工作原理是:对写和读都是直接操作工作主存的。
2. synchronized 关键字
  • synchronized关键字是用来控制线程同步的,就是在多线程的环境下,控制synchronized代码段不被多个线程同时执行,是一种阻塞性的锁,synchronized既可以加在一段代码上,也可以加在方法上。
  • synchronized(this)及非static的synchronized方法,只能防止多个线程同时执行同一个对象的同步代码段。当synchronized锁住一个对象后,别的线程如果也想拿到这个对象的锁,就必须等待这个线程执行完成释放锁,才能再次给对象加锁,这样才达到线程同步的目的。即使两个不同的代码段,都要锁同一个对象,那么这两个代码段也不能在多线程环境下同时运行。所以我们在用synchronized关键字的时候,能缩小代码段的范围就尽量缩小,能在代码段上加同步就不要再整个方法上加同步。这叫减小锁的粒度,使代码更大程度的并发。
  • 原因是基于以上的思想,锁的代码段太长了,别的线程是不是要等很久。如果用synchronized加在静态方法上,就相当于用××××.class锁住整个方法内的代码块,此时是锁住该类的Class对象,相当于一个全局锁。
  • 使用synchronized修饰的方法或者代码块可以看成是一个原子操作。一个线程执行互斥代码过程如下:
  1. 获得同步锁;
  2. 清空工作内存;
  3. 从主内存拷贝对象副本到工作内存;
  4. 执行代码(计算或者输出等);
  5. 刷新主内存数据;
  6. 释放同步锁。
    所以,synchronized既保证了多线程的并发有序性,又保证了多线程的内存可见性。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,699评论 6 513
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,124评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,127评论 0 358
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,342评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,356评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,057评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,654评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,572评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,095评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,205评论 3 339
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,343评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,015评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,704评论 3 332
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,196评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,320评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,690评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,348评论 2 358

推荐阅读更多精彩内容