【Java进阶营】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() 方法,该方法直接返回单例中的示例。在此我向大家推荐一个架构学习交流圈。交流学习指导伪鑫:1253431195(里面有大量的面试题及答案)里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多
然后在执行 Test.main() 方法
执行结果如下:

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

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值