前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 序列化 之 单例模式

Java 序列化 之 单例模式

作者头像
java404
发布2018-05-18 15:07:53
7560
发布2018-05-18 15:07:53
举报
文章被收录于专栏:java 成神之路java 成神之路

序列化相关文章:

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

示例一

User 类

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

代码语言:javascript
复制
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 的单例实例序列化后在反序列化。

代码语言:javascript
复制
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() 方法

代码语言:javascript
复制
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()中返回的对象直接替换在反序列化过程中创建的对象。


本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018.04.10 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 示例一
    • User 类
      • Test 类
      • readResolve() 方法
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档