“避免序列化灾难:掌握实现 Serializable 的真相!(二)”

文章目录

      • 一、什么是序列化?
      • 二、`Serializable` 是如何起作用的?
      • 三、为什么不自动序列化所有对象?
      • 四、Java 序列化的底层原理
        • 序列化的核心步骤:
      • 五、反序列化的原理
      • 六、总结:为什么必须实现 `Serializable` 才能序列化?
      • 推荐阅读文章

在学习 Java 的时候,你可能遇到过一个疑问: 为什么类必须实现 Serializable,否则就没法序列化对象? 这个看似简单的要求,其实和 Java 的底层机制紧密相关。今天,我们今天就来聊聊这个问题的本质,带你了解为什么 Serializable 是 Java 序列化必不可少的一环。

一、什么是序列化?

先来简单回顾一下什么是序列化。序列化的作用,就是把一个Java 对象转换成字节流,这样我们就可以把它存到文件、通过网络传输、或者传递给其他系统。反过来,把字节流还原成对象的过程叫“反序列化”。

例如,你有一个 User 对象,包含用户名和年龄。序列化后,这个对象会变成一串字节:

User user = new User("Alice", 25);
// 序列化对象到文件
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("user.ser"));
out.writeObject(user);
out.close();

之后你可以读取文件,反序列化回一个 User 对象:

ObjectInputStream in = new ObjectInputStream(new FileInputStream("user.ser"));
User user = (User) in.readObject();
in.close();

这个过程依赖的是 Java 的序列化机制,而 Serializable 就是这个机制的核心。

二、Serializable 是如何起作用的?

Serializable 是一个标记接口,它本身没有定义任何方法。那它为什么如此重要?让我们从 Java 的序列化机制来看。

Java 序列化机制背后是由 ObjectOutputStreamObjectInputStream 这两个类来完成的。当你调用 writeObject() 方法时,Java 会执行一系列操作来把对象转换为字节流。而这些操作,都是基于对象的类是否实现了 Serializable 接口。如果类没有实现 Serializable,Java 会直接抛出异常,阻止序列化的进行。

为什么会这样呢?这是因为 Java 底层的序列化机制需要知道一个类是否允许被序列化。Serializable 就像是给 Java 的一个“信号”,告诉它这个类的对象是可以被安全地序列化和反序列化的。如果类没有实现 Serializable,Java 默认认为这个类的对象不能被序列化,也就没有办法执行后续操作。

三、为什么不自动序列化所有对象?

你可能会想:为什么 Java 不让所有的类自动支持序列化呢?这样岂不是更方便?

实际上,Java 不自动序列化所有对象,是出于安全性和效率的考虑。原因如下:

  1. 安全性问题
    并不是所有类的对象都适合被序列化。有些类可能包含敏感信息,比如密码、系统配置、用户身份信息等。如果随便把这些类序列化,可能会带来安全隐患。例如,一些类设计时可能没有考虑到数据暴露的问题,而序列化可能会无意间将这些数据泄露给外部系统。

  2. 效率问题
    序列化和反序列化的过程是需要消耗系统资源的,特别是在涉及到复杂对象时。如果 Java 默认让所有类都支持序列化,不仅会增加不必要的开销,还可能导致性能下降。比如某些类可能有非常大的对象图,序列化这些对象可能会产生很大的字节流,影响系统效率。

因此,Java 让开发者自己决定哪些类需要支持序列化,而不是一刀切地自动支持所有类。通过让类实现 Serializable,开发者明确地告诉 Java:这个类是可以被序列化的

四、Java 序列化的底层原理

当一个对象实现了 Serializable 接口,Java 底层会使用反射机制来遍历对象的所有字段,并将这些字段的值按顺序转换成字节流。这其中包括:

  • 基本类型(如 intdouble 等)的直接序列化。
  • 对象类型的递归序列化(比如一个对象里包含另一个对象,也会序列化这个嵌套的对象)。

Java 在序列化时,不仅仅是保存对象的字段值,还会保存一些元信息,包括:

  • 类的名称
  • 类的版本号(serialVersionUID
  • 对象的字段及其类型

这些信息使得在反序列化时,Java 能够找到正确的类,并根据保存的字段值恢复出对象。如果类发生了变化(比如字段修改了),还可以通过 serialVersionUID 来判断序列化和反序列化是否兼容。

序列化的核心步骤:
  1. 判断对象是否实现 Serializable:只有实现了这个接口的对象,才能继续序列化。
  2. 遍历对象的字段:使用反射,Java 会依次获取对象的每个字段,并将其转换成字节流。
  3. 保存对象的元数据:保存类的信息、版本号、以及字段的类型和值。
  4. 生成字节流:最终将这些信息“打包”成字节流,便于存储或传输。

五、反序列化的原理

反序列化的过程则是将字节流重新解析成对象。Java 会根据字节流中的元数据,找到对应的类,并使用反射机制重建对象的实例。这个过程需要类的 serialVersionUID 与字节流中的版本号一致,确保类结构没有发生不兼容的变化。

如果对象的类没有实现 Serializable,反序列化时也会失败,因为 Java 没有办法恢复出一个未标记为可序列化的类的对象。

六、总结:为什么必须实现 Serializable 才能序列化?

  • Serializable 是一个标记,告诉 Java 这个类可以安全地被序列化和反序列化。Java 需要这个信号来决定是否执行序列化操作。
  • 安全性:并不是所有对象都适合被序列化,Serializable 让开发者有选择地开放序列化权限,避免敏感数据泄露。
  • 效率:序列化和反序列化是有成本的,不自动支持所有类序列化可以避免不必要的性能开销。
  • 底层机制:Java 使用 Serializable 接口配合反射来实现序列化和反序列化过程,只有实现了 Serializable 的类,Java 才会执行序列化流程。

因此,Java 之所以要求类实现 Serializable,是为了通过这一标记来确保序列化的安全性、性能,以及在底层能够正确执行序列化过程。这也是为什么我们必须显式地让类实现 Serializable,否则 Java 序列化机制就没法发挥作用。

推荐阅读文章

  • 使用 Spring 框架构建 MVC 应用程序:初学者教程
  • 有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
  • 如何理解应用 Java 多线程与并发编程?
  • Java Spring 中常用的 @PostConstruct 注解使用总结
  • 线程 vs 虚拟线程:深入理解及区别
  • 深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
  • 10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
  • “打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”
  • Java 中消除 If-else 技巧总结
  • 线程池的核心参数配置(仅供参考)
  • 【人工智能】聊聊Transformer,深度学习的一股清流(13)
  • Java 枚举的几个常用技巧,你可以试着用用
  • 如何理解线程安全这个概念?
  • 理解 Java 桥接方法
  • Spring 整合嵌入式 Tomcat 容器
  • Tomcat 如何加载 SpringMVC 组件

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/57398.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

最好的ppt模板网站是哪个?做PPT不可错过的18个网站!

现在有很多PPT模板网站,但真正免费且高质量的不多,今天我就分享主流的国内外PPT模板下载网站,并且会详细分析这些网站的优缺点,这些网站都是基于个人实际使用经验的,免费站点会特别标注,让你可以放心下载&a…

【Jmeter】jmeter指定jdk版本启动

背景: 因权限问题,不能修改操作系统的环境变量或者因jmeter启动加载的默认jdk8版本低,需要指定jdk XX版本启动Jmeter 解决办法: 进入jmeter bin目录选择jmeter.bat,记事本编辑jmeter.bat, 在最前面添加 set MINIMAL_…

go 中的斐波那契数实现以及效率比较

package mainimport ("fmt""math/big""time" )// FibonacciRecursive 使用递归方法计算斐波那契数列的第n个数 func FibonacciRecursive(n int) *big.Int {if n < 1 {return big.NewInt(int64(n))}return new(big.Int).Add(FibonacciRecursiv…

抗肺癌市场迎新突破,十款创新药物获批!

在肺癌治疗领域&#xff0c;近年来取得了令人瞩目的进展。随着科学技术的不断进步和临床研究的深入&#xff0c;多款创新肺癌药物相继获批上市&#xff0c;为患者带来了前所未有的治疗选择和希望。本文将详细介绍十款最新获批的肺癌创新药物&#xff0c;探讨它们的特点、适应症…

Leetcode 1135. 最低成本连通所有城市

1.题目基本信息 1.1.题目描述 想象一下你是个城市基建规划者&#xff0c;地图上有 n 座城市&#xff0c;它们按以 1 到 n 的次序编号。 给你整数 n 和一个数组 conections&#xff0c;其中 connections[i] [x_i, y_i, cost_i] 表示将城市 x_i 和城市 y_i 连接所要的cost_i&…

Stable Diffusion Web UI 大白话术语解释 (二)

归纳整理&#xff0c;Stable Diffusion Web UI 使用过程中&#xff0c;相关术语 ControlNet ControlNet 说简单点&#xff0c;就是你可以给 AI 一些“规则”&#xff0c;比如让它根据某些线条、结构或者骨架去画图。 这样能让 AI 画出更符合你要求的图片&#xff0c;特别适合画…

买华为系的车,这个理由无法拒绝

文 | AUTO芯球 作者 | 雷慢 激动人心啊&#xff0c; 超过45万问界车主&#xff0c; 还有几十万其他用华为鸿蒙智能座舱系统的车主, 大家的软件安全、自主可控问题&#xff0c; 这下可以彻底放心了&#xff01; 为什么&#xff0c;就在昨晚&#xff0c; 完全自主可控、彻…

ECharts饼图-饼图34,附视频讲解与代码下载

引言&#xff1a; 在数据可视化的世界里&#xff0c;ECharts凭借其丰富的图表类型和强大的配置能力&#xff0c;成为了众多开发者的首选。今天&#xff0c;我将带大家一起实现一个饼图图表&#xff0c;通过该图表我们可以直观地展示和分析数据。此外&#xff0c;我还将提供详…

可编辑60页PPT | 大数据基础知识培训课件

荐言分享&#xff1a;在当今信息化高速发展的时代&#xff0c;大数据已成为推动各行各业创新转型的关键力量。无论是金融、医疗、教育还是零售等行业&#xff0c;大数据的应用都为企业带来了前所未有的机遇和挑战。为了帮助学员更好地理解和应用大数据&#xff0c;我们精心设计…

鸿蒙应用的Tabs 组件怎么使用

鸿蒙应用中的Tabs组件是一个用于通过页签进行内容视图切换的容器组件&#xff0c;每个页签对应一个内容视图。以下是Tabs组件的使用方法&#xff1a; 一、基本结构 Tabs组件的页面组成包含两个部分&#xff0c;分别是TabContent和TabBar。TabContent是内容页&#xff0c;TabB…

纯css实现瀑布流! 附源码!!!

瀑布流用于展示图片信息,我这里用的背景颜色来代替图片 PC端效果 源码(直接复制粘贴就可以运行了!!!) <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>PC端瀑布流</title><style>.box {w…

Umi UI报错:连接失败,请尝试重启dev服务

Umi UI连接失败&#xff0c;请尝试重启dev服务 使用umi ui时遇到以下问题 报错如下 从报错可以看出是淘宝镜像失效的问题&#xff0c;检查淘宝镜像 可以看出淘宝镜像是最新的&#xff0c;并无问题 经过查找发现报错是因为依赖文件中使用了旧的淘宝镜像&#xff0c;在node…

2025国内10大主流免费在线客服系统

ttkefu在线客服 多渠道接入&#xff1a;支持网站、APP、社交媒体等多种渠道接入&#xff0c;方便客户随时进行咨询。多样化沟通&#xff1a;提供图文、视频、表情等多种消息类型&#xff0c;提升沟通效率。智能客服与人工客服结合&#xff1a;机器人客服能够自动识别并理解用户…

美国FDA注册和FDA检测的区别

FDA注册 FDA注册是美国食品药品管理局对进入美国市场的产品进行企业和产品信息登记的过程&#xff0c;其目的主要包含反恐和限制不符合要求产品的市场准入&#xff0c;FDA注册主要针对的企业主要有&#xff1a;食品类企业&#xff08;包含所欲可食用产品及动物饲料&#xff09;…

100种算法【Python版】第2篇——分治法

分而治之 1 分治法原理2 示例说明:归并排序2.1 分治法的步骤2.2 归并排序代码3 分治法应用3.1 最近点对问题3.1.1 Python3代码3.1.2 分治法思路说明3.2 快速傅里叶变换(FFT)3.2.1 Python3代码3.2.1 分治法思路说明3.3 最长公共子序列问题3.3.1 Python3代码3.3.2 分治法思路说…

Java全栈经典面试题剖析4】JavaSE高级 -- 包装类,String, 类方法

目录 面试题3.1 什么是自动装箱与拆箱&#xff1f;用什么方式来装箱与拆箱&#xff1f; 面试题3.2 int和Integer有什么区别&#xff1f; 面试题3.3 Integer常量池 面试题3.4 字符串常量池 面试题3.5 这句代码创建了几个对象? String str1 new String("xyz");…

【大数据应用开发】2023年全国职业院校技能大赛赛题第10套

如有需要备赛资料和远程培训,可私博主,详细了解 目录 任务A:大数据平台搭建(容器环境)(15分) 任务B:离线数据处理(25分) 任务C:数据挖掘(10分) 任务D:数据采集与实时计算(20分) 任务E:数据可视化(15分) 任务F:综合分析(10分) 任务A:大数据平台搭…

RootNeighboursDataset(helpers.dataset_classes文件中的root_neighbours_dataset.py)

任务类型:回归 用途:在 `RootNeighboursDataset` 中,任务是给定一棵根树,预测根节点度数为6的邻居的特征平均值。因此,模型需要基于根节点的结构,找到度为6的邻居,并计算其特征的平均值。这属于回归问题,因为目标是预测连续值(特征的平均值)。 from helpers.dataset_…

C++ 抛异常

目录 一.抛异常与运行崩溃的区别 1.运行崩溃 2.抛异常 二.抛异常机制存在的意义 1.清晰的处理错误 2.结构化的错误管理 3.跨函数传递错误信息 4.异常对象多态性 三.抛异常的使用方法 1.抛出异常 (throw) 2.捕获异常 (catch) 3.标准异常类 四.抛异常的处理机制 1.抛…

【MySQL备份】Percona XtraBackup

这份文档针对的是最新发布的版本&#xff1a;Percona XtraBackup 2.4.29&#xff08;发布说明&#xff09;。 Percona XtraBackup是一款针对MySQL系列服务器的开源热备份工具&#xff0c;在备份过程中不会锁定您的数据库。它能够对MySQL 5.1、5.5、5.6和5.7服务器以及带有Xtra…