惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

S
Schneier on Security
Hugging Face - Blog
Hugging Face - Blog
V
Visual Studio Blog
博客园 - Franky
酷 壳 – CoolShell
酷 壳 – CoolShell
Last Week in AI
Last Week in AI
博客园 - 叶小钗
博客园_首页
阮一峰的网络日志
阮一峰的网络日志
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Application and Cybersecurity Blog
Application and Cybersecurity Blog
TaoSecurity Blog
TaoSecurity Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
J
Java Code Geeks
爱范儿
爱范儿
宝玉的分享
宝玉的分享
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
量子位
N
News and Events Feed by Topic
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Recent Commits to openclaw:main
Recent Commits to openclaw:main
SecWiki News
SecWiki News
MyScale Blog
MyScale Blog
AI
AI
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
博客园 - 【当耐特】
Security Archives - TechRepublic
Security Archives - TechRepublic
F
Fortinet All Blogs
V2EX - 技术
V2EX - 技术
T
Troy Hunt's Blog
有赞技术团队
有赞技术团队
W
WeLiveSecurity
Project Zero
Project Zero
T
Tor Project blog
Help Net Security
Help Net Security
L
LINUX DO - 最新话题
IT之家
IT之家
The Hacker News
The Hacker News
腾讯CDC
Schneier on Security
Schneier on Security
N
News and Events Feed by Topic
C
Cisco Blogs
博客园 - 聂微东
Webroot Blog
Webroot Blog
Forbes - Security
Forbes - Security
M
MIT News - Artificial intelligence
C
Cyber Attacks, Cyber Crime and Cyber Security
雷峰网
雷峰网
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
A
About on SuperTechFans

博客园_首页

Plist 二进制格式 Milvus 和 PGVector,哪个更好? OpenClaw 已过时?在 VS Code 中运行 Hermes Agent! 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent UIImageView 设置图片不生效的原因排查 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里借助MU实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 前端小白一句话,AI 帮我做了个颜值拉满的桌面媒体播放器。当代码不再是门槛,一句话编程就是现实。 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 Claude code入门 - 陈彦斌 一文学习入门 ThingsBoard 开源物联网平台 GitHub 热门项目 | 2026年04月16日 如何为GIT设置全局勾子,为每次提交追加信息 Number.isFinite和isFinite与isNaN()和Number.isNaN的区别 PortSwigger SQL注入LAB2 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) GLM模型这么火,咱们用vllm也咧一个呗! 深入理解 AbortController:从底层原理到跨语言设计哲学 字符串学习笔记 多租户系统框架的基础模块设计和分析设计 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 ElasticSearch7.X版本配置密码 用Manim实现动态交点计算--从一个动点问题说起 团结引擎+Addressable+Instant Game打包抖音小游戏 function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 bubseek —— 让 Agent 的足迹,变成团队的洞察 通过 C# 读取并导出 PDF 书签 如何用 GitHub Actions 实现 Steam 自动化发布 【从0到1构建一个ClaudeAgent】并发-后台任务 .NET 高级开发 | 定制 ASP.NET Core 框架 电子小白:什么是运算放大器(运放) zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 工作的意义、工作的困难认知再思考 .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 开了 TUN 模式还是直连?90% 的人都踩过这个坑 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术
Java中的序列化和反序列化:让你的对象学会“变身术”
佛祖让我来巡山 · 2026-06-24 · via 博客园_首页

前言:一个让你“卡壳”的面试题

先问大家一个问题:面试官问你“什么是Java序列化”,你会怎么回答?

是不是脑海里蹦出几个关键词——“把对象存到文件里”“实现Serializable接口”——然后就卡壳了?

别慌。今天咱们就花一篇文章的时间,把序列化和反序列化彻底聊透。不整那些晦涩难懂的技术黑话,咱们从最接地气的角度,一步步揭开它的面纱。

一、序列化是什么?——对象的“过安检”

想象一个场景:你要坐飞机,随身带的行李不能直接拎上飞机,得先过安检——从“三维实体”变成“X光图像”,扫描确认安全后才能放行。

在Java的世界里,对象就是你的行李,而网络传输、文件存储、Redis缓存这些“通道”就是安检机。它们不认内存里的Java对象,只认字节流。所以我们必须先把对象“过一遍安检”——转换成字节序列(byte[]) ——这个过程,就叫序列化(Serialization)

一句话总结:序列化 = 把内存中的Java对象 → 转成字节流(可存储、可传输)

那反序列化呢?当这串字节流到达目的地(比如另一个服务、本地文件或者Redis),接收方需要把它还原成原来的Java对象才能继续使用。这个“从字节流变回对象”的过程,就叫反序列化(Deserialization)

反序列化 = 把字节流 → 还原成内存中的Java对象

打个比方:序列化就像把对象“冷冻”起来,反序列化就是“解冻”,解冻之后还是原来那个对象,状态一点没变。

二、为什么要序列化?——存在的意义

好,概念搞清楚了。但你可能要问:我为什么要费这个劲儿把对象转来转去?直接拿对象用不香吗?

问题在于,Java对象是活在内存里的。程序一关、一断电,内存里的对象就灰飞烟灭了。序列化就是来解决这个问题的——它给了对象“永生”的能力。

具体来说,序列化的价值体现在三个场景:

1. 数据持久化:把对象保存到硬盘文件或数据库里,程序重启后还能恢复。比如游戏存档,就是把你的游戏角色序列化存下来,下次打开游戏再反序列化回去。

2. 网络传输:两个服务之间要传对象数据,必须转成二进制流才能在网上跑。RPC、微服务调用,底层都离不开序列化。

3. 缓存存储:把对象存到Redis、Memcached这类缓存里,本质也是先序列化再存储。

可以说,没有序列化,分布式系统就玩不转,数据持久化也无从谈起。

三、怎么实现序列化?——三分钟上手

理论说完了,咱们直接上代码。

第一步:实现Serializable接口

在Java中,想让一个类的对象能被序列化,只需要让这个类实现java.io.Serializable接口:

import java.io.Serializable;

public class Person implements Serializable {
    private static final long serialVersionUID = 1L;
    private String name;
    private int age;
    
    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
    
    @Override
    public String toString() {
        return "Person{name='" + name + "', age=" + age + "}";
    }
}

注意,Serializable接口里面什么方法都没有,它只是一个标记接口(Marker Interface) ——就像衣服上的标签,告诉JVM:“这个类的对象可以序列化”。

如果某个类没实现这个接口就强行序列化,JVM会毫不客气地抛出NotSerializableException

第二步:用ObjectOutputStream序列化

有了可序列化的类,接下来用ObjectOutputStream把它写到文件里:

import java.io.*;

public class SerializationDemo {
    public static void main(String[] args) {
        Person person = new Person("张三", 25);
        
        // 序列化:对象 → 文件
        try (ObjectOutputStream oos = new ObjectOutputStream(
                new FileOutputStream("person.ser"))) {
            oos.writeObject(person);
            System.out.println("序列化成功!");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

writeObject()方法就是核心,它负责把对象变成字节流并写入输出流。

第三步:用ObjectInputStream反序列化

序列化存下来的文件,怎么把它变回对象?用ObjectInputStream

public class DeserializationDemo {
    public static void main(String[] args) {
        // 反序列化:文件 → 对象
        try (ObjectInputStream ois = new ObjectInputStream(
                new FileInputStream("person.ser"))) {
            Person person = (Person) ois.readObject();
            System.out.println("反序列化成功:" + person);
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

readObject()方法从字节流中读取数据并重建对象。注意它返回的是Object类型,需要强制类型转换

整个流程就是三步:实现Serializable → writeObject写入 → readObject读出

四、深入原理:序列化背后发生了什么?

会用只是第一步,咱们再往底层挖一挖,看看序列化到底是怎么工作的。

当你调用writeObject()时,JVM会做以下几件事:

  1. 检查对象是否实现了Serializable接口
  2. 生成类的序列化描述符(包括类名、serialVersionUID、所有字段信息等)
  3. 递归序列化对象的所有非transient、非static字段——如果字段里还引用了其他对象,就继续序列化那个对象
  4. 把所有字节数据写入输出流

反序列化时,JVM不会调用对象的构造函数,而是直接通过反射设置字段值。这也是为什么反序列化出来的对象看起来像“凭空变出来”的——它确实没用new

另外,序列化只保存对象的字段值类元信息(类名、字段名、类型等),不保存静态变量、方法信息和构造函数。

五、serialVersionUID:对象的“身份证”

这是面试里的高频考点,咱们单独拎出来讲。

每个实现了Serializable的类,都可以定义一个serialVersionUID

private static final long serialVersionUID = 1L;

这个ID是做什么用的?简单说,它是这个类的版本号

当你序列化一个对象时,serialVersionUID会随着对象数据一起被写入字节流。反序列化时,JVM会把字节流里的ID和当前类的ID进行比对:

  • 一致 → 放心反序列化
  • 不一致 → 抛出InvalidClassException

那问题来了:如果不写会怎样?

如果你不显式声明,JVM会根据类的结构(字段名、类型、方法等)自动计算一个。但这就埋下了一个隐患——你哪天给类加了一个字段,JVM算出来的ID就变了,之前序列化好的数据就反序列化不回来了

所以最佳实践是:永远显式声明serialVersionUID。这样就算类结构变了,只要ID不变,反序列化时新增的字段会被设为默认值(null或0),删除的字段会被忽略,至少不会直接崩溃。

六、transient:有些字段我不想存

不是所有的字段都值得存。比如密码这种敏感信息,或者临时计算出来的数据,序列化的时候最好跳过。

这时候就用上transient关键字了:

public class User implements Serializable {
    private static final long serialVersionUID = 1L;
    private String username;
    private transient String password;  // 这个字段不序列化
    private transient int tempCache;    // 临时数据也不存
}

transient修饰的字段,序列化时会被直接忽略,反序列化时会被赋上默认值(引用类型为null,基本类型为0或false)。

七、原生序列化的优缺点

Java原生序列化(通过Serializable)好用是好用,但也有它的局限性:

优点:

  • 简单,JDK自带,零依赖
  • 自动处理对象图(循环引用、对象引用都能处理好)

缺点:

  • 体积大:生成的字节流包含大量元信息,浪费存储和带宽
  • 速度慢:性能比不上很多第三方方案
  • 不跨语言:Java序列化的字节流,Python、Go这些语言读不懂

所以在实际项目中,很多人会选择JSON(Jackson、Gson)、Protocol BuffersKryo等替代方案。尤其是需要跨语言通信的场景,JSON或Protobuf几乎是必选项。

八、安全警告:别随便反序列化!

最后说一个严肃的问题:反序列化有安全风险

如果你反序列化的数据来自不可信来源(比如用户上传的文件、网络请求传来的数据),攻击者可能精心构造恶意字节流,在反序列化时执行任意代码。

黄金法则:永远不要反序列化来自不可信来源的数据

从Java 9开始,可以用ObjectInputFilter来限制允许反序列化的类集合,相当于给反序列化加了一道白名单。

结语

回顾一下今天的内容:

  • 序列化 = 对象 → 字节流(过安检)
  • 反序列化 = 字节流 → 对象(复活术)
  • 实现方式 = 实现Serializable + ObjectOutputStream/ObjectInputStream
  • 版本控制 = 显式声明serialVersionUID
  • 跳过字段 = 用transient关键字
  • 注意事项 = 性能一般、不跨语言、注意安全

序列化是Java里一个看似简单但内涵丰富的知识点。它让对象可以“走出”内存,在文件系统和网络之间自由穿梭——没有它,很多我们习以为常的功能(缓存、RPC、持久化)都无从谈起。

希望这篇文章能帮你彻底搞懂序列化和反序列化。下次面试官再问你,可就不只是说“把对象存到文件里”那么简单了。


有什么问题或者想深入探讨的点,欢迎在评论区留言!