登录
首页 >  文章 >  java教程

Java对象序列化详解:ObjectInputStream与ObjectOutputStream使用方法

时间:2026-01-29 14:05:36 391浏览 收藏

golang学习网今天将给大家带来《Java对象序列化:ObjectInputStream与ObjectOutputStream使用详解》,感兴趣的朋友请继续看下去吧!以下内容将会涉及到等等知识点,如果你是正在学习文章或者已经是大佬级别了,都非常欢迎也希望大家都能给我建议评论哈~希望能帮助到大家!

ObjectInputStream/ObjectOutputStream读写对象需严格满足序列化契约:类须实现Serializable,所有非transient非static字段类型也须可序列化;必须成对使用缓冲流;读写顺序严格一致;serialVersionUID缺失或变更、字段修改等导致运行时异常。

在Java中如何使用ObjectInputStream与ObjectOutputStream_Java对象序列化解析

Java 的 ObjectInputStreamObjectOutputStream 能直接读写对象,但前提是对象必须严格满足序列化契约——否则运行时抛 java.io.NotSerializableException 或反序列化后字段为 null/0,不是“不工作”,而是“ silently fail”。

对象必须实现 Serializable 接口且字段可访问

仅声明 implements Serializable 不够,所有非 transientstatic 字段所属类型也得可序列化。常见踩坑点:

  • ArrayListString 等 JDK 类已实现 Serializable,放心用
  • 自定义类(如 User)若含第三方库对象(如 org.json.JSONObject),而该类没实现 Serializable,就会报错
  • 未显式定义 private static final long serialVersionUID = 1L;,JVM 自动生成的 UID 在类结构变动后会不一致,导致 InvalidClassException
  • transient 字段不会被序列化,反序列化后为默认值(null0false

流必须成对使用,且顺序严格一致

ObjectOutputStream 写入顺序,必须和 ObjectInputStream 读取顺序完全一致。不能跳过、重排或重复读——没有“按字段名读取”机制,纯靠字节流位置。

  • 先写 out.writeInt(100),再写 out.writeObject(new Date()),那读的时候必须先 in.readInt(),再 in.readObject()
  • 如果写入了多个对象,每次 readObject() 拿到的是一个完整对象,不是“下一条记录”
  • 多次调用 writeObject() 后关闭流,再用新 ObjectInputStream 打开文件,仍能按序读出全部对象;但若中途异常中断,可能损坏流头,后续读取失败

文件或网络传输前务必包装为 BufferedXXXStream

直接用 FileInputStream / FileOutputStream 构造 ObjectInputStream / ObjectOutputStream 会导致性能极差,且某些场景(如 socket)易出 StreamCorruptedException

  • 必须用 new ObjectInputStream(new BufferedInputStream(new FileInputStream("a.bin"))),不能省略 BufferedInputStream
  • 同理,写端必须用 new ObjectOutputStream(new BufferedOutputStream(new FileOutputStream("a.bin")))
  • Socket 场景下,客户端和服务端的流构造顺序必须镜像:服务端先 new ObjectOutputStream(out),再 new ObjectInputStream(in);客户端反之,否则握手失败
try (ObjectOutputStream oos = new ObjectOutputStream(
        new BufferedOutputStream(new FileOutputStream("user.dat")))) {
    oos.writeObject(new User("Alice", 30));
    oos.writeObject(new User("Bob", 25));
} catch (IOException e) {
    e.printStackTrace();
}

try (ObjectInputStream ois = new ObjectInputStream(
        new BufferedInputStream(new FileInputStream("user.dat")))) {
    User u1 = (User) ois.readObject(); // Alice
    User u2 = (User) ois.readObject(); // Bob
} catch (IOException | ClassNotFoundException e) {
    e.printStackTrace();
}

真正麻烦的不是语法,是版本兼容性:serialVersionUID 变了、字段删了、父类没序列化、用了 Lambda 表达式作为字段值——这些都不会在编译时报错,只会在反序列化那一行突然崩掉。

好了,本文到此结束,带大家了解了《Java对象序列化详解:ObjectInputStream与ObjectOutputStream使用方法》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>