个人随笔之单例模式

为什么使用单例模式?

确保某一个类只有一个实例,避免产生多个对象消耗过多的资源, 如要访问IO和 数据库等资源

实现单例模式的几个关键点:

构造函数不对外开放,一般为private
通过一个静态方法或枚举返回单例类对象
确保单例类的对象只有一个,尤其是在多线程环境下
确保单例类对象反序列化时不会重新构建对象

饿汉模式—空间换取时间,线程安全

public class Singleton {

    private Singleton() { }

    private static Singleton instance = new Singleton();

    public static Singleton getInstance() {
        return instance;
    }
}

懒汉模式

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

在getinstance方法中加入了synchronized关键字,也就是getInstance是一个同步方法,是线程安全的,但是这种方式存在性能上的缺陷,每次调用getInstance都会进行同步,消耗不必要的资源

优点:只有在使用时才会被实例化,在一定程度上节约了资源
缺点:第一次加载时需要及时实例化,反应稍慢,最大问题是每次调用getInstance都进行同步,造成不必要的同步开销

Double check lock 模式

能够在需要时才实例化,保证线程安全,只有在第一次调用时才进行同步,以后调用getInstance都不会进行同步锁

public class Singleton {

    private static Singleton instance = null;

    private Singleton() {}

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

getInstance方法中对instance进行了两次判空,第一次主要是为了避免不必要的同步,第二层则是为了在null的情况下创建实例

instance = new Singleton();此操作并不是一个原子操作,这句代码最终会被编译成多条汇编指令

  1. 给Singleton的实例分配内存
  2. 调用Singleton的构造函数,初始化成员字段
  3. 将instance对象指向分配的内存空间(此时instance就不是null了)

由于Java编译器允许处理器乱序执行,第2,3条的指令执行顺序是无法保证的,执行顺序可能为1-2-3或1-3-2,假定A线程执行的顺序为1-3-2,当执行到第三条指令时,此时instance已经不为null,切换到B线程,判断instance不为空,直接取走了instance,使用时就会报错,double check lock模式此时就失效了。

在Java1.5以后,具体化了volatile关键字,volatile会将修改的变量值立即更新到主存中,保证变量的可见性,只需将instance定义改为private volatile static Singleton instance = null,就可以保证instance对象每次都是从主内存中读取。

优点:资源利用率高,第一次执行getInstance时单例对象才会被实例化
缺点:第一次加载时反应稍慢

静态内部类模式

public class Singleton {

    private Singleton() {}

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

第一次加载Singleton类时并不会初始化instance,只有在第一次调用getInstance方法时才会初始化instance实例,因此,第一次调用getInstance方法会触发虚拟机加载SingletonHolder类,这种方式不仅能保证线程安全,也能保证实例对象的唯一性,同时也延迟了单例的实例化,推荐用此模式

枚举单例

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

枚举在Java中和普通类是一样的,不仅能够拥有字段,还能有自己的方法,最重要的是默认枚举实例的创建是线程安全的,并且在任何情况下它都是一个单例。

在上述几种单例模式的实现中,在反序列化的情况下他们会出现重新穿件对象,枚举方式则不会

通过序列化可以将一个单例的实例对象写到磁盘,然后再读回来,从而有效的获得一个实例。即使构造函数是私有的,反序列化时依然可以通过特殊的途径去创建类的一个新实例。反序列化操作提供了一个很特别的钩子函数readResolve,如果要杜绝单例对象在被反序列化时重新生成对象,那么必须加入readResolve函数,也就是在readResolve函数中将单例对象返回,而不是重新生成一个新对象。

private Object readResolve() {
    return SingletonHolder.instance;
}
public class Singleton implements Serializable {

    private Singleton() {
    }

    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }

    private static class SingletonHolder {
        private static Singleton instance = new Singleton();
    }

//    private Object readResolve() {
//        return SingletonHolder.instance;
//    }


    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Singleton s = Singleton.getInstance();
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.obj"));
        oos.writeObject(s);
        oos.flush();
        oos.close();

        FileInputStream fis = new FileInputStream("Singleton.obj");
        ObjectInputStream ois = new ObjectInputStream(fis);
        Singleton s1 = (Singleton) ois.readObject();
        ois.close();
        System.out.println(s + "\n" + s1);
    }
}

//output
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa
com.miracle.thirdlibsourcecode.Singleton@6acbcfc0

不加钩子函数反序列化的时候则会创建新的对象

public class Singleton implements Serializable {

    private Singleton() {
    }

    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }

    private static class SingletonHolder {
        private static Singleton instance = new Singleton();
    }

    private Object readResolve() {
        return SingletonHolder.instance;
    }


    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Singleton s = Singleton.getInstance();
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.obj"));
        oos.writeObject(s);
        oos.flush();
        oos.close();

        FileInputStream fis = new FileInputStream("Singleton.obj");
        ObjectInputStream ois = new ObjectInputStream(fis);
        Singleton s1 = (Singleton) ois.readObject();
        ois.close();
        System.out.println(s + "\n" + s1);
    }
}

//output
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa
com.miracle.thirdlibsourcecode.Singleton@1d44bcfa

加了钩子函数readResolve以后,反序列化就不会重新创建对象了

而对于枚举,并不存在这个问题,因为即使反序列化也不会生成新的实例对象

通过对enum反编译发现其实enum是继承了Enum抽象类,在反序列化时,反射在通过newInstance创建对象时,会检查该类是否ENUM修饰,如果是则抛出异常,反射失败。

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

推荐阅读更多精彩内容