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

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 等更灵活的格式。
http://www.xdnf.cn/news/6470.html

相关文章:

  • 奇妙协同效应,EtherNet IP与PROFINET网关优化半导体生产线
  • Git .gitattributes 文件用途详解
  • Baklib知识中台驱动智能服务新实践
  • ZCC6303x-60V/1.2MHz 高效率升压 LED 恒流驱动替代SY7301
  • 【图片识别工具】批量单据识别批量重命名,批量OCR识别图片文字并重命名,批量改名工具的使用步骤和注意事项
  • Modbus TCP转Profinet网关:数字化工厂异构网络融合的核心枢纽
  • pciutils-3.5.5-win64工具的使用方法
  • Java大师成长计划之第23天:Spring生态与微服务架构之服务发现与注册中心
  • 使用命令行拉取 Git 仓库
  • 数学复习笔记 9
  • 自学嵌入式 day 18 - 数据结构 1
  • 嵌软面试每日一阅----FreeRTOS
  • SpringBoot实现简单的API代理服务器
  • Sumsub 活体检测与人证对比 Java Demo
  • pytorch训练可视化工具---TensorBoard
  • Linux 防火墙 firewalld 实战配置教程!
  • 将.pt文件执行图像比对
  • Java详解RabbitMQ工作模式之发布订阅模式
  • 具备AI功能的银河麒麟桌面操作系统已正式上市
  • 手搓传染病模型(SEI - SEIAR )
  • xp_cmdshell bcp 导出文件
  • 道通龙鱼系列-混合翼无人机:垂直起降+长时续航
  • 嵌入式自学第二十二天(5.15)
  • 02、基础入门-Spring生态圈
  • 云上玩转 Qwen3 系列之三:PAI-LangStudio x Hologres构建ChatBI数据分析Agent应用
  • 机器学习第十三讲:独热编码 → 把“红黄蓝“颜色变成001/010/100的数字格式
  • 数据结构之图的应用场景及其代码
  • MySQL 用户权限管理:从入门到精通
  • 26考研 | 王道 | 计算机组成原理 | 一、计算机系统概述
  • Java:跨越时代的编程语言传奇