美文网首页
从单例模式到序列化

从单例模式到序列化

作者: 捉虫大师 | 来源:发表于2018-08-11 23:26 被阅读17次

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

  • 构造方法私有化
  • 静态方法返回实例
  • 当心多线程会创建多个实例
  • 小心反序列化时会创建多个实例
方法一:类初始化的时候就创建
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;
    }

}

相关文章

  • 枚举单例——避免反序列化破坏单例

    六种单例模式实现 枚举单例 深度解析单例与序列化

  • 深度解析单例与序列化之间的爱恨情仇~

    本文将通过实例+阅读Java源码的方式介绍序列化是如何破坏单例模式的,以及如何避免序列化对单例的破坏。 单例模式,...

  • 单例与序列化的那些事儿

    本文将通过实例+阅读Java源码的方式介绍序列化是如何破坏单例模式的,以及如何避免序列化对单例的破坏。 单例模式,...

  • 从单例模式到序列化

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

  • java程序的性能优化(二)

    善用设计模式 单例模式:各类写法,反序列化破坏单例 代理模式:jdk接口代理,asm代理,cglib,javass...

  • 单例模式序列化时注意!

    单例模式类实现Serializable接口后, 在序列化时, getInstance方法变的不可用, 所以单例模式...

  • ABAP和Java单例模式的攻防

    ABAP 通过序列化/反序列化攻击单例模式: 绕过了单例的限制,构造了第二个实例。 Java 除了用序列化/反序列...

  • 单例模式安全之序列化攻击

    单例模式安全之序列化攻击 源码 什么是序列化攻击呢? 简单说,一个单例对象经过序列化再反序列化后,内存中会存在两个...

  • 设计模式-单例模式【实现、序列化、反射】

    设计模式-单例模式【实现、序列化、反射】 [toc] 1. 实现 单例模式的实现有很多种,分类方式也不一而足,比如...

  • 单例模式

    由于性能问题,优化,采用双重检查锁 懒汉式内部类单例 枚举类从JDK层面就保证不能被序列化和反射所破坏单例模式 枚...

网友评论

      本文标题:从单例模式到序列化

      本文链接:https://www.haomeiwen.com/subject/xdrjbftx.html