Java 序列化 之 单例模式

序列化相关文章:

当我们使用Singleton模式时,应该是期望某个类的实例应该是唯一的,但如果该类是可序列化的,那么发序列化后还会是单例的吗?下面我们通过如下示例一来验证一下:

示例一

User 类

User 类是单例模式,使用的饿汉模式,在类加载的时候就创建对象实例。

public class User implements Serializable {
    private static final long serialVersionUID = 3380014540967816490L;

    private String userName;
    private String password;

    private static User user = new User("zhangsan", "test");
    
    private User(String userName, String password) {
        this.userName = userName;
        this.password = password;
    }
    public static User getInstance() {
        return user;
    }
    public String getUserName() {
        return userName;
    }
    public String getPassword() {
        return password;
    }

Test 类

测试类,把 User 的单例实例序列化后在反序列化。

public class Test{
    public static void main(String[] args) throws Exception {
        File file = new File("d:\\a.user");
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(file));
        oos.writeObject(User.getInstance());


        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
        User user = (User) ois.readObject();
        System.out.println(user);
        
        if(user==User.getInstance()){
            System.out.println("同一个实例");
        }else{
            System.out.println("不同的实例");
        }
    }
}

执行结果如下:

输出的结果: User [userName=zhangsan, password=123456] 不同的实例

通过结果可以看出,单例模式的饿汉模式也无法确保对象实例是单例的。

那么我们应该怎么解决这个问题呢?

readResolve() 方法

public class User implements Serializable {

    private static final long serialVersionUID = 3380014540967816490L;

    private String userName;
    private String password;

    private static User user = new User("zhangsan", "123456");

    private User(String userName, String password) {
        this.userName = userName;
        this.password = password;
    }
    public static User getInstance() {
        return user;
    }
    public String getUserName() {
        return userName;
    }
    public String getPassword() {
        return password;
    }
    public Object readResolve(){
        return getInstance();
    }
    @Override
    public String toString() {
        return "User [userName=" + userName + ", password=" + password + "]";
    }
}

我们在 User 类中添加了一个 readResolve() 方法,该方法直接返回单例中的示例。 然后在执行 Test.main() 方法 执行结果如下:

输出的结果: User [userName=zhangsan, password=123456] 相同的实例

无论是实现Serializable接口,或是Externalizable接口,当从I/O流中读取对象时,readResolve()方法都会被调用到。实际上就是用readResolve()中返回的对象直接替换在反序列化过程中创建的对象。


本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java帮帮-微信公众号-技术文章全总结

Java基础19(02)总结IO流,异常try…catch,throws,File类

案例: a:考试成绩必须满足在0——100之间,不满足就抛出异常。 package cn.itcast_08(1); import java.util.Sca...

3975
来自专栏IT可乐

Java IO详解(一)------File 类

File 类:文件和目录路径名的抽象表示。 注意:File 类只能操作文件的属性,文件的内容是不能操作的。 1、File 类的字段 ?   我们知道,各个平台之...

2179
来自专栏JAVA高级架构

Java设计模式--单例模式

2294
来自专栏小二的折腾日记

day5(面向对象2)

用来将文件或文件夹封装成对象。 方便对文件与文件夹的属性信息进行操作。 File对象可以作为参数传递给

651
来自专栏Linux驱动

22.QT-QXmlStreamReader解析,QXmlStreamWriter写入

XML 用于存储数据,数据的形式类似于树结构(参考: http://www.runoob.com/xml/)

1435
来自专栏difcareer的技术笔记

JNI实现源码分析【三 间接引用表】正文0x01: IndirectRefTable0x02: 作用域0x03: jobject到Object的映射0x04: JNI在背后默默做的事

在JNI实现源码分析【二 数据结构】的参数传递一节中,我们提到,JNI为了安全性的考虑使用了形如jobject的结构来传递参数。而jobject被表述为指针,但...

912
来自专栏企鹅号快讯

mysql 字段时间类型的比较

字段的时间类型分为: ,,,,; 下面就分别介绍这几种时间类型的区别 ? 每个时间类型都有一个有效范围和一个零值,当指定的类型的值超过有效范围时,就会使用零值 ...

2108
来自专栏我的博客

Zend_Db_Adapter使用详情

Zend_Db_Adapter是zendfrmaeword的数据库抽象层api. 基于pdo, 你可以使用 Zend_Db_Adapter 连接和处理多种 数据...

3574
来自专栏ml

strcpy和memcpy的区别

strcpy和memcpy都是标准C库函数,它们有下面的特点。 strcpy提供了字符串的复制。即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还...

3346
来自专栏用户画像

JAVA笔试题

4.public、private、protected、Friendly的区别与作用域,以及不写时的区别?

1031

扫码关注云+社区

领取腾讯云代金券