序列化和反序列化

一、核心概念(用生活例子理解)

想象一下:

  • 你有一个乐高拼好的小汽车(对应程序中的Java 对象 ,比如ChatMessage)。
  • 序列化:把这个乐高小汽车拆成一个个独立的零件,并用清单记录每个零件的位置、形状(→ 把 Java 对象转换成字节流 / JSON 字符串等可存储 / 传输的格式)。
  • 反序列化:根据清单,把零散的乐高零件重新拼成原来的小汽车(→ 把字节流 / JSON 字符串还原成原来的 Java 对象)。
官方定义

表格

操作 核心含义
序列化 (Serialize) 内存中的对象 (如ChatMessage实例)转换为字节序列 / 字符串(如 JSON),方便存储到文件 / 数据库、或网络传输。
反序列化 (Deserialize) 存储 / 传输的字节序列 / 字符串 (如数据库里的 JSON)还原为内存中的对象,让程序可以直接使用。

二、为什么需要序列化 / 反序列化?

以你之前的代码场景为例:

  1. 存储需求ChatMessage对象是内存中的临时数据,程序重启就会消失。把它序列化成 JSON 字符串存到 MongoDB,就能持久化保存。
  2. 传输需求:如果你的聊天程序是分布式的(比如前端→后端→数据库),对象无法直接在网络中传输,需要序列化成 JSON / 二进制流,传输后再反序列化。
  3. 跨语言交互:JSON 是通用格式,Java 序列化的 JSON 字符串,前端 JavaScript 可以反序列化使用,实现跨语言数据交互。

三、代码示例(直观理解)

以你熟悉的ChatMessage为例,用 Jackson 库演示序列化 / 反序列化:

1. 准备实体类

java

运行

复制代码
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.Data;

@Data // 简化getter/setter,需要lombok依赖
public class ChatMessage {
    private String memoryId;
    private String content;
    private Long timestamp;
}
2. 序列化(对象 → JSON 字符串)

java

运行

复制代码
public class SerializeDemo {
    public static void main(String[] args) throws Exception {
        // 1. 创建内存中的对象
        ChatMessage message = new ChatMessage();
        message.setMemoryId("mem_123");
        message.setContent("你好,MongoDB!");
        message.setTimestamp(System.currentTimeMillis());

        // 2. 序列化:对象 → JSON字符串(相当于拆乐高)
        ObjectMapper objectMapper = new ObjectMapper();
        String jsonStr = objectMapper.writeValueAsString(message);

        System.out.println("序列化后的JSON:");
        System.out.println(jsonStr);
        // 输出:{"memoryId":"mem_123","content":"你好,MongoDB!","timestamp":1740000000000}
    }
}
3. 反序列化(JSON 字符串 → 对象)

java

运行

复制代码
public class DeserializeDemo {
    public static void main(String[] args) throws Exception {
        // 1. 从数据库/文件中读取的JSON字符串
        String jsonStr = "{\"memoryId\":\"mem_123\",\"content\":\"你好,MongoDB!\",\"timestamp\":1740000000000}";

        // 2. 反序列化:JSON字符串 → 对象(相当于拼乐高)
        ObjectMapper objectMapper = new ObjectMapper();
        ChatMessage message = objectMapper.readValue(jsonStr, ChatMessage.class);

        System.out.println("反序列化后的对象:");
        System.out.println("memoryId: " + message.getMemoryId()); // 输出:mem_123
        System.out.println("content: " + message.getContent());   // 输出:你好,MongoDB!
    }
}

四、常见的序列化格式

  1. JSON :最常用,可读性高、跨语言,你的代码中就是用这种格式(chatMessages.getContent() 是 JSON 字符串)。
  2. XML:老牌格式,可读性高但体积大,现在用得少。
  3. 二进制序列化(Java 原生):把对象转成字节流,体积小但只能 Java 之间使用,不跨语言。
  4. Protobuf:谷歌出品,二进制格式,体积小、效率高,适合高性能传输场景。

总结

  1. 序列化:对象 → 字符串 / 字节流(便于存储 / 传输),核心是 "拆"。
  2. 反序列化:字符串 / 字节流 → 对象(便于程序使用),核心是 "拼"。
  3. 你代码中的 messagesFromJson() 就是典型的反序列化操作,把 MongoDB 中存储的 JSON 字符串还原成ChatMessage对象。
相关推荐
程序员Sunday5 小时前
说点不一样的。GPT-5.3 与 Claude Opus 4.6 同时炸场,前端变天了?
前端·gpt·状态模式
前端不太难6 小时前
HarmonyOS 游戏项目,从 Demo 到可上线要跨过哪些坑
游戏·状态模式·harmonyos
万物得其道者成10 小时前
阿里云 H5 一键登录接入实战:前后端完整实现
阿里云·云计算·状态模式
前端不太难10 小时前
在 HarmonyOS 上,游戏状态该怎么“死而复生”
游戏·状态模式·harmonyos
木斯佳20 小时前
前端八股文面经大全:26届秋招滴滴校招前端一面面经-事件循环题解析
前端·状态模式
hepingfly21 小时前
不再单打独斗!用 Agent Teams 让 7 个 Claude 同时帮你开发
状态模式
C澒1 天前
Remesh 框架详解:基于 CQRS 的前端领域驱动设计方案
前端·架构·前端框架·状态模式
前端不太难1 天前
HarmonyOS 游戏里,Ability 是如何被重建的
游戏·状态模式·harmonyos
程序员agions1 天前
2026年,微前端终于“死“了
前端·状态模式