Java序列化和反序列化(未完成,待8.27更新)

简介: 序列化:Java对象转换为字节序列。反序列化:字节序列转换为Java对象。 实现了Serializable或Externalnalizable接口的接口或类,通过ObjectInputStream或ObjectOutputStream可以进行二进制文件与Java对象之间的转换。
  • 序列化:Java对象转换为字节序列
  • 反序列化:字节序列转换为Java对象

实现了Serializable或Externalnalizable接口的接口或类,通过ObjectInputStream或ObjectOutputStream可以进行二进制文件与Java对象之间的转换。

Serializable

  • 实现了Serializable的对象,可以将该对象的非transient的实例变量进行序列化或反序列化。
  • 若重写了 readObject(ObjectInputStream in)writeObject(ObjectOutputSteam out) 方法,则会调用这两个方法进行序列化或反序列化。

PS:non-transient字段不会参与序列化会反序列化,这样,有时传输数据对象可以节省不必要的时空开销。当然,如果实现了 readObject()writeObject() 方法,仍有机会在这两个方法内序列化和反序列化trasient字段。

普通序列化

MyObject obj = new MyObject("zou", "male");

// 序列化
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("/user/zoujs/documents/java.obj"));
out.writeObject(obj);
out.close();

// 反序列化
ObjectInputStream in = new ObjectInputStream(new FileOutputStream("/user/zoujs/documents/java.obj"));
MyObject object = (MyObject)in.readObject();
in.close();

结合从前写的Closeable的文章,可以将代码写成如下形式。

// 序列化
MyObject obj = new MyObject("zou", "male");
try (ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("/users/zoujs/documents/java.obj"))){
    out.writeObject(obj);
}

// 反序列化
MyObject object;
try (ObjectInputStream in = new ObjectInputStream(new FileInputStream("/users/zoujs/documents/java.obj"))){
    object = (MyObject)in.readObject();
}


readObject()writeObject()

这两个方法可以在序列化和反序列化时定制用户的行为,即使它们被声明为private,但这丝毫不影响它们的作用。

public class MyObject implements Serializable {
    private static final long serialVersionUID = 7906828868467432175L;
    
    private transient String name;
    // 此处省略其他属性以及 setter和getter

    private void writeObject(ObjectOutputStream oos) throws Exception{
        oos.defaultWriteObject();
        oos.writeObject(name);
        System.out.println("writeObject");
    }

    private void readObject(ObjectInputStream ois) throws Exception{
        ois.defaultReadObject();
        name = (String)ois.readObject();
        System.out.println("readObject");
    }
}

主函数

MyObject obj = new MyObject("zou", "male");
try (ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("/users/zoujs/documents/java.obj"))){
    out.writeObject(obj);
}

// 反序列化
MyObject object;
try (ObjectInputStream in = new ObjectInputStream(new FileInputStream("/users/zoujs/documents/java.obj"))){
    object = (MyObject)in.readObject();
}

我们发现, object的name属性虽然是已经用transient关键字修饰,但我们依然从最后的object对象读到了设置过的属性。

此外,我们发现这两个方法使用private修饰,那么读写流是如何调用到这两个方法的呢?

实际上,ObjectInputStream和ObjectOutputStream通过反射来寻找对象的这两个私有方法的,所以必须用private来修饰。


writeReplace()

除writeObject外,还有一种自定义序列化方式, 它可以做到在序列化某一对象时改为序列化另一个对象。

Java能够保证在序列化之前调用writeReplace()方法,如果该方法返回另一个Java对象,则系统转为序列化另一个对象。

下面,我们在MyObject这个bean中添加下面的函数,

private Object writeReplace() {
    AnotherObject obj = new AnotherObject();
    obj.setName("name");
    return obj;
}

并且,我们在流读入的时候将读到的对象转换为AnotherObject

AnotherObject obj;
try (ObjectInputStream in = new ObjectInputStream(new FileInputStream("/users/zoujs/documents/java.obj"))){
    obj = (AnotherObject) in.readObject();
}
System.out.println(obj.getName());

结果:name

writeObject()明明写入了一个MyObject对象,但是真正序列化的却是AnotherObject对象。并且,MyObject这个bean中同时包含writeObject()readObject()writeReplace() 三个函数时依然如此。

由此我们得知,系统在序列化某个对象前,只要这个对象包含 writeReplace() 函数,就会去调用它来序列化。如果该方法返回另一个对象,则对另一个对象采取相同操作,直到不返回对象为止。程序是否进行自身的 writeObject()readObject() 要看它是否包含 writeReplace() 函数。

相关文章
|
2月前
|
JSON 数据格式 索引
Python中序列化/反序列化JSON格式的数据
【11月更文挑战第4天】本文介绍了 Python 中使用 `json` 模块进行序列化和反序列化的操作。序列化是指将 Python 对象(如字典、列表)转换为 JSON 字符串,主要使用 `json.dumps` 方法。示例包括基本的字典和列表序列化,以及自定义类的序列化。反序列化则是将 JSON 字符串转换回 Python 对象,使用 `json.loads` 方法。文中还提供了具体的代码示例,展示了如何处理不同类型的 Python 对象。
|
2月前
|
存储 安全 Java
🌟Java零基础-反序列化:从入门到精通
【10月更文挑战第21天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
95 5
|
2月前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
36 3
|
2月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
3月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
3月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第3天】在Java编程的世界里,对象序列化与反序列化是实现数据持久化和网络传输的关键技术。本文将深入探讨Java序列化的原理、应用场景以及如何通过代码示例实现对象的序列化与反序列化过程。从基础概念到实践操作,我们将一步步揭示这一技术的魅力所在。
|
3月前
|
消息中间件 存储 Java
大数据-58 Kafka 高级特性 消息发送02-自定义序列化器、自定义分区器 Java代码实现
大数据-58 Kafka 高级特性 消息发送02-自定义序列化器、自定义分区器 Java代码实现
73 3
|
2月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
63 0
|
5月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
4月前
|
存储 Java 开发者
Java编程中的对象序列化与反序列化
【9月更文挑战第20天】在本文中,我们将探索Java编程中的一个核心概念——对象序列化与反序列化。通过简单易懂的语言和直观的代码示例,你将学会如何将对象状态保存为字节流,以及如何从字节流恢复对象状态。这不仅有助于理解Java中的I/O机制,还能提升你的数据持久化能力。准备好让你的Java技能更上一层楼了吗?让我们开始吧!