从单例模式到序列化

单例模式大家应该都不陌生,只要写过一些代码的,估计天天用,好处也自然不多说,有些对象创建一个就够了,比如数据库或者网络的连接,创建一个就够了。实现单例模式需要做到:

  • 构造方法私有化
  • 静态方法返回实例
  • 当心多线程会创建多个实例
  • 小心反序列化时会创建多个实例
方法一:类初始化的时候就创建
package disignpattern;

public class Singleton1 {

    private static Singleton1 instance = new Singleton1();

    private Singleton1() {

    }

    private static Singleton1 getInstance() {
        return instance;
    }
}

这种方式简单粗暴,但是缺点是用不到的时候也可能初始化。

方法二:使用的时候创建
package disignpattern;

public class Singleton2 {

    private static Singleton2 instance = null;

    private Singleton2() {

    }

    private synchronized static Singleton2 getInstance() {
       if (instance == null) {
           instance = new Singleton2();
       }
        return instance;
    }

}

这里需要注意多线程的问题,需要将初始化的代码加一个锁,不过这里这种方式是一种粗粒度的加锁方式,效率不是很高,也可以这么写,效果一样:

package disignpattern;

public class Singleton3 {

    private static Singleton3 instance = null;

    private Singleton3() {

    }

    private static Singleton3 getInstance() {

        synchronized (Singleton3.class) {
            if (instance == null) {
                instance = new Singleton3();
            }
        }
        return instance;
    }
}

网上有一种针对这个的优化版本:

package disignpattern;

public class Singleton4 {

    private static volatile Singleton4 instance = null;

    private Singleton4() {

    }

    private static Singleton4 getInstance() {

        if (instance == null) {
            synchronized (Singleton4.class) {
                if (instance == null) {
                    instance = new Singleton4();
                }
            }
        }
        
        return instance;
    }

}

这个方法有一个高大上的名词叫双重校验锁,instance被volatile修饰说明instance任何时候拿到的都是最新的值(可见性),所以如果instance不是null(大部分时候)都不需要进入锁,因为锁比较耗时,所以这样是一个优化。

方式三:使用枚举创建
package disignpattern;

public enum Singleton5 {

    INSTANCE;

    public static Singleton5 getInstance() {
        return INSTANCE;
    }

}

这也是《java编程思想》里面推荐的单例模式实现方式,实现非常简单。

方法四:内部静态类
package disignpattern;

public class Singleton6 {

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

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

}

这个方式算是方法一的升级版,由于内部类在外部类被加载的时候不会立即被加载,只有当getInstance被调用时才加载内部类,所以算是对方法一的一个延迟版本。

一开始说到的反序列化是什么回事呢?
我们看一段代码:

package disignpattern;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

public class Singleton1 implements Serializable {

    private static final long sericlVersionUID = 1L;

    private static Singleton1 instance = new Singleton1();

    private Singleton1() {

    }

    private static Singleton1 getInstance() {
        return instance;
    }

    public static void main(String[] args) throws Exception {
        FileOutputStream fileOutputStream = null;
        ObjectOutputStream objectOutputStream = null;
        try {
            fileOutputStream = new FileOutputStream("/tmp/Singleton");
            objectOutputStream = new ObjectOutputStream(fileOutputStream);
            objectOutputStream.writeObject(instance);
        } finally {
            objectOutputStream.flush();
            objectOutputStream.close();
            fileOutputStream.close();
        }

        FileInputStream fileInputStream = null;
        ObjectInputStream objectInputStream = null;
        Singleton1 s = null;
        try {
            fileInputStream = new FileInputStream("/tmp/Singleton");
            objectInputStream = new ObjectInputStream(fileInputStream);
            s = (Singleton1) objectInputStream.readObject();
        } finally {
            objectInputStream.close();
            fileInputStream.close();
        }
        System.out.println( s == instance);
    }

}

这个输出false,也就是说反序列化的时候单例模式失效了,这是为啥?底层原理我看了网上资料说是反射,没有去读源码来深究,想想反射是可以实现的,这也是为啥单例模式会失效了,所以推荐的是枚举模式,枚举就算反射也没法创建多个实例。针对这个问题,我们可以通过在单例的类中实现一个readResolve方法就行,至于原理,我们下回分析序列化的时候可以一起分析一下。

package disignpattern;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

public class Singleton1 implements Serializable {

    private static final long sericlVersionUID = 1L;

    private static Singleton1 instance = new Singleton1();

    private Singleton1() {

    }

    private static Singleton1 getInstance() {
        return instance;
    }

    public static void main(String[] args) throws Exception {
        FileOutputStream fileOutputStream = null;
        ObjectOutputStream objectOutputStream = null;
        try {
            fileOutputStream = new FileOutputStream("/tmp/Singleton");
            objectOutputStream = new ObjectOutputStream(fileOutputStream);
            objectOutputStream.writeObject(instance);
        } finally {
            objectOutputStream.flush();
            objectOutputStream.close();
            fileOutputStream.close();
        }

        FileInputStream fileInputStream = null;
        ObjectInputStream objectInputStream = null;
        Singleton1 s = null;
        try {
            fileInputStream = new FileInputStream("/tmp/Singleton");
            objectInputStream = new ObjectInputStream(fileInputStream);
            s = (Singleton1) objectInputStream.readObject();
        } finally {
            objectInputStream.close();
            fileInputStream.close();
        }
        System.out.println( s == instance);
    }

    private Object readResolve() {
        return instance;
    }

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

推荐阅读更多精彩内容

  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,253评论 4 34
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,652评论 18 139
  • 1.单例模式概述 (1)引言 单例模式是应用最广的模式之一,也是23种设计模式中最基本的一个。本文旨在总结通过Ja...
    曹丰斌阅读 2,915评论 6 47
  • 前言 本文主要参考 那些年,我们一起写过的“单例模式”。 何为单例模式? 顾名思义,单例模式就是保证一个类仅有一个...
    tandeneck阅读 2,510评论 1 8
  • 2017-9-24 星期天 晴 读经内容:庄子01 老子1-10 少年儿童诗词启蒙01-06 今天5点10分起床,...
    菜问妈妈阅读 180评论 0 0