9.序列化

序列化

问题背景:

  1. 代码修改:将实体类字段 long id 改为 Long id
  2. 序列化机制:使用 JDK 序列化存储到 Redis
  3. 反序列化失败:重启服务后读取 Redis 数据时抛出 InvalidClassException

根本原因:
JDK 序列化依赖 类结构完整性,字段类型变更(基本类型→包装类型)导致类结构变化,触发序列化 ID 变化。


二、序列化 ID(serialVersionUID)的核心作用

场景结果
未声明 serialVersionUIDJVM 根据类结构自动生成 ID
✅ 类结构不变:反序列化成功
❌ 类结构变化(如 longLong):自动生成的 ID 改变 → 反序列化失败
显式声明 serialVersionUID手动固定版本号
✅ 类结构变化(如增删字段):可反序列化(需处理字段兼容)
字段类型变更:二进制结构不匹配 → 反序列化失败

📌 关键结论:
即使显式声明 serialVersionUID,修改字段类型(如 longLong)仍会导致二进制数据不兼容!


三、序列化协议对比(JDK vs JSON)

特性JDK 序列化JSON 序列化
数据格式二进制(不可读)文本(Key-Value 结构,可读)
兼容性强依赖类结构(类型严格匹配)弱依赖(仅需字段名匹配,类型可自动转换)
跨语言仅支持 Java多语言通用
空间效率高(二进制压缩)低(文本冗余)
CPU 开销高(压缩/解压复杂)低(文本解析简单)
类型变更容忍度❌ 不允许(如 longLong 报错)✅ 允许(自动装箱/拆箱)

四、解决方案

1. 兼容性修改原则
  • 禁止直接修改字段类型:如必须修改,需通过数据迁移重新序列化历史数据。
  • 显式声明 serialVersionUID:避免因方法/非关键字段变更导致意外失败。
2. 序列化协议选型
场景推荐方案
纯 Java 服务、高压缩需求JDK 序列化(需严格兼容)
多语言交互、字段易变性高JSON / Protocol Buffers
容忍字段类型变更(如 longLong强制使用 JSON
3. 补救措施
  • 数据迁移:修改字段类型后,删除旧序列化数据或编写脚本转换历史数据。
  • 双写策略:新字段用新名称(如 idLong),逐步迁移数据。

五、总结

  1. JDK 序列化是“脆弱”的
    longLong 二进制结构不同,类型变更必然导致反序列化失败。
  2. JSON 更适应迭代
    通过文本结构和自动类型转换,天然兼容字段类型升级。
  3. 安全迭代建议
    // 显式声明 serialVersionUID 是底线,但非万能!
    public class User implements Serializable {
        private static final long serialVersionUID = 1L; // 固定版本号
        private Long id; // 优先使用包装类型(兼容 null)
    }
    
    高兼容性系统:直接使用 JSON 序列化(如 Spring Data Redis 默认 JSON 序列化器)。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值