当前位置: 首页 > news >正文

Java 序列化(Serialization)

一、理论说明

1. 序列化的定义

Java 序列化是指将对象转换为字节流的过程,以便将其存储到文件、数据库或通过网络传输。反序列化则是将字节流重新转换为对象的过程。通过实现java.io.Serializable接口,类可以被标记为可序列化的,该接口是一个标记接口,不包含任何方法。

2. 核心用途

  • 对象持久化:将对象状态保存到文件或数据库,例如缓存会话信息。
  • 远程通信:在网络中传输对象,例如 RMI(远程方法调用)。
  • 深拷贝:通过序列化和反序列化实现对象的深拷贝。

二、实现序列化

1. 可序列化类

必须实现Serializable接口,并定义serialVersionUID(推荐)。

import java.io.Serializable;public class User implements Serializable {private static final long serialVersionUID = 1L; // 版本控制private String name;private int age;// 构造方法、getter/setter 略
}

2. 序列化过程

使用ObjectOutputStream将对象写入流。

try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("user.ser"))) {User user = new User("Alice", 30);oos.writeObject(user); // 序列化对象
} catch (IOException e) {e.printStackTrace();
}

3. 反序列化过程

使用ObjectInputStream从流读取对象。

try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("user.ser"))) {User user = (User) ois.readObject(); // 反序列化对象System.out.println(user.getName()); // 输出: Alice
} catch (IOException | ClassNotFoundException e) {e.printStackTrace();
}

三、关键特性

1. serialVersionUID

  • 作用:确保序列化和反序列化时类的版本一致性。若版本号不匹配,会抛出InvalidClassException
  • 生成方式
    private static final long serialVersionUID = 42L; // 手动指定
    // 或通过 IDE 自动生成(基于类结构的哈希值)

    2. 瞬态字段(transient

    transient修饰的字段不会被序列化,例如敏感信息或临时数据。

    private transient String password; // 不会被序列化

    3. 自定义序列化逻辑

    通过重写writeObject()readObject()方法,可以自定义序列化过程。

    private void writeObject(ObjectOutputStream out) throws IOException {out.defaultWriteObject(); // 调用默认序列化out.writeUTF(name); // 自定义处理
    }private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {in.defaultReadObject(); // 调用默认反序列化name = in.readUTF(); // 自定义处理
    }

    四、注意事项

    1. 版本兼容性

  • 修改类结构(如添加字段)可能导致serialVersionUID不匹配,需谨慎管理版本号。
  • 2. 安全风险

  • 反序列化时可能执行恶意代码(如 RCE 漏洞),建议只反序列化可信来源的数据。
  • 3. 替代方案

  • 对于复杂场景,可使用 JSON/XML 等跨语言格式(如 Gson、Jackson)替代原生序列化。
  • 五、应用实例

    1. 对象深拷贝

    public static <T> T deepCopy(T obj) {try (ByteArrayOutputStream baos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(baos)) {oos.writeObject(obj);try (ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bais)) {return (T) ois.readObject();}} catch (IOException | ClassNotFoundException e) {throw new RuntimeException(e);}
    }

    2. 序列化集合

    List<User> users = Arrays.asList(new User("Alice", 25),new User("Bob", 30)
    );// 序列化集合
    try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("users.ser"))) {oos.writeObject(users);
    }

  • 六、面试题

  • 题目:
  • 答案:
  • 七、自我总结

  • Java 序列化提供了一种便捷的对象持久化和传输机制,但需注意:

  • 确保类实现Serializable接口并定义serialVersionUID
  • 使用transient关键字排除敏感字段。
  • 谨慎处理版本兼容性和安全风险。
  • 在跨语言场景中,考虑使用 JSON/XML 等更灵活的格式。

相关文章:

  • MUSE Pi Pro 使用TiTanTools烧录镜像
  • SiFli-SDK 编译
  • 车载诊断进阶篇 --- 车载诊断概念
  • 基于互联网和LabVIEW的多通道数据采集系统仿真设计
  • Spring 模拟转账开发实战
  • Spring MVC HttpMessageConverter 的作用是什么?
  • 2025年,如何制作并部署一个完整的个人博客网站
  • 智能视觉赋能精准抓取:富唯智能重新定义机械臂应用新高度​
  • 嵌入式学习笔记 D21:双向链表的基本操作
  • 深入浅出:Windows系统DLL劫持提权原理
  • 位运算题目:找到最接近目标值的函数值
  • 面试从微前端拓展到iframe是如何通信的
  • 计量——检验与代理变量
  • 【备忘踩坑】Android单元测试中读取测试assets的方法
  • EdgeShard:通过协作边缘计算实现高效的 LLM 推理
  • NY244NY249美光闪存颗粒NY252NY256
  • U9C与钉钉审批流对接完整过程
  • 数据结构—排序(斐波那契数列,冒泡,选择,插入,快速,归并,图,广度优先算法)
  • 【Docker】Docker安装Redis
  • Git命令总结
  • 钕铁硼永磁材料龙头瞄准人形机器人,正海磁材:已向下游客户完成小批量供货
  • 俄外交部:俄乌伊斯坦布尔谈判改在当地时间15日下午举行
  • 回望星河深处,唤醒文物记忆——读《发现武王墩》
  • 晋级四强!WTA1000罗马站:郑钦文2比0萨巴伦卡
  • 国务院关税税则委员会关于调整对原产于美国的进口商品加征关税措施的公告
  • 秦洪看盘|交易新逻辑,银行股成A股稳定器