深入探究序列化与反序列化:原理、应用和最佳实践

2023-09-22 15:19:04

什么是对象的序列化和反序列化

序列化(Serialization)是指将对象转化为字节流的过程,以便于存储或通过网络进行传输。
反序列化(Deserialization)则是将字节流转化为对象的过程,恢复原始对象的状态。

在计算机科学中,序列化和反序列化是常用的数据处理技术,用于在不同系统、不同编程语言之间传递对象或持久化对象的状态。
在这里插入图片描述

序列化步骤

1.创建输出流:将对象写入到输出流中。
2.对象编码:将对象的数据转化为字节流的形式,可以使用不同的编码方式(如二进制编码、JSON、XML等)。
3.输出到目标:将编码后的字节流输出到目标位置,如文件、内存、网络等。

反序列化步骤

1.创建输入流:从输入流中读取字节流。
2.对象解码:将字节流解码为原始对象的数据形式。
3.构建对象:使用解码后的数据构建对象,并恢复对象的状态。

案例演示

在Java中,序列化和反序列化是通过实现Serializable接口来实现的。
要进行序列化,需要按以下步骤进行操作:
确保类实现了java.io.Serializable接口。
创建一个输出流(如java.io.FileOutputStream)将对象写入文件或网络流中。
创建一个java.io.ObjectOutputStream对象,将其连接到输出流上。
使用ObjectOutputStream的writeObject()方法将对象写入输出流。

示例代码如下所示:

public class SerializationExample {
    public static void main(String[] args) {
        MyClass obj = new MyClass(); // 要进行序列化的对象

        try {
            FileOutputStream fileOut = new FileOutputStream("data.ser"); // 序列化的目标文件
            ObjectOutputStream out = new ObjectOutputStream(fileOut);
            out.writeObject(obj); // 将对象写入输出流
            out.close();
            fileOut.close();
            System.out.println("对象已被序列化并保存为 data.ser");
        } catch(IOException e) {
            e.printStackTrace();
        }
    }
}

要进行反序列化,需要按照以下步骤操作:
1.创建一个输入流(如java.io.FileInputStream)读取序列化的对象。
2.创建一个java.io.ObjectInputStream对象,将其连接到输入流上。
3.使用ObjectInputStream的readObject()方法从输入流中读取对象,返回一个Object类型的对象。
将返回的对象转换为需要的类型(例如,强制类型转换为具体的类)。
示例代码如下所示:

public class DeserializationExample {
    public static void main(String[] args) {
        MyClass obj = null;
        
        try {
            FileInputStream fileIn = new FileInputStream("data.ser"); // 读取序列化的文件
            ObjectInputStream in = new ObjectInputStream(fileIn);
            obj = (MyClass) in.readObject(); // 从输入流中读取对象并转换类型
            in.close();
            fileIn.close();
        } catch(IOException e) {
            e.printStackTrace();
            return;
        } catch(ClassNotFoundException e) {
            e.printStackTrace();
            return;
        }
        
        System.out.println("对象已成功反序列化");
        // 对反序列化后的对象进行操作
        // ...
    }
}

Java中哪些字段不能序列化

在Java中,有一些字段是不能被序列化的。这些字段包括:

  • 静态变量(static variables):静态变量属于类级别,而不是实例级别。它们不包含在任何特定的对象中,因此不能被序列化。
  • transient变量(transient variables):使用transient关键字修饰的变量不会被序列化。transient变量通常用于表示临时状态或敏感信息,因此在序列化过程中会被忽略。
  • 方法(methods):在Java中,方法是不能被序列化的。只有对象的数据状态才能被序列化和反序列化。
  • 匿名内部类和局部内部类(anonymous inner classes and local inner classes):匿名内部类和局部内部类都包含对外部类的引用,这会导致序列化时的问题。
  • 类型为java.util.function包中的函数式接口类型。由于函数式接口通常具有lambda表达式或方法引用,因此它们无法被序列化。

需要注意的是,如果一个类实现了Serializable接口,但其中包含不可序列化的字段,则在序列化该类的实例时,这些字段的值将被忽略。如果需要对这些字段进行序列化,可以通过自定义序列化过程来实现。

为了避免某些字段被序列化,可以使用transient关键字修饰这些字段,让它们在序列化过程中被忽略。例如:

public class MyClass implements Serializable {
    private transient int transientField;  // transient字段,在序列化时被忽略

    // 其他字段和方法...
}

在使用transient关键字控制字段的序列化行为时,有几个方面需要注意:
序列化版本兼容性:当你对类进行修改时,特别是涉及到需要序列化的字段时,需要注意序列化版本的兼容性。如果对已序列化的对象进行反序列化时,版本不兼容可能会导致异常或数据丢失。建议在类中添加一个serialVersionUID字段,并在进行类的修改时适当更新它,以确保反序列化时的版本兼容性。

序列化与反序列化的重要性

  • 可实现跨平台和跨语言的数据交换,使得不同系统之间可以共享数据。
  • 可进行数据持久化,将对象保存到存储介质中,以便再次读取和使用。
  • 可用于远程调用,将对象通过网络传输到远程系统,并在远程系统上还原为对象进行处理。
  • 在不同的编程语言中,通常都会提供相应的序列化和反序列化机制或库,以便开发者方便地实现对象的序列化和反序列化操作。

序列化与反序列化的应用场景

  • 敏感数据的安全控制:在使用transient关键字标记字段时,请确保其中不包含敏感信息,例如密码、密钥等。被transient修饰的字段在对象被序列化后将被忽略,这样可以避免敏感数据泄露。
  • 自定义序列化逻辑:某些情况下,你可能需要自定义对象的序列化和反序列化逻辑。可以通过实现writeObject()和readObject()方法来实现自定义的序列化过程。在这种情况下,即使字段被transient修饰,你也可以在这些方法中手动控制字段的序列化行为。
  • 跨平台兼容性:如果你的应用程序需要在不同的平台上运行或与其他语言进行交互,务必注意跨平台兼容性问题。某些平台或语言可能对transient关键字的处理方式有所不同,因此在进行跨平台或跨语言的序列化操作时需要格外小心。

总之,在使用transient关键字时,要仔细考虑对象的序列化行为以及对应的框架的特性。确保版本兼容性、安全控制和跨平台兼容性,以确保序列化操作的正确性和稳定性。

更多推荐

GB28181协议-SDP详解

SDP协议SDP全称是SessionDescriptionProtocol,翻译过来就是描述会话的协议。主要用于两个会话实体之间的媒体协商。SDP描述由许多文本行组成,文本行的格式为<类型>=<值>,表示为key=value;SIP负责建立和释放会话,一般来说,会话中包含相关的媒体,比如视频和音频。媒体数据是由SDP描

别问怎么下载,金蝶云星空SaaS BI系统不用下载

国产自研的奥威软件-金蝶云星空SaaSBI,不下载不安装,从浏览器上一键注册登录即可使用:一键点击下载金蝶云星空方案,执行后,BI系统将基于金蝶云星空内的数据与方案自带的BI报表,智能计算分析指标,生成数十张BI数据可视化分析报表。奥威-金蝶云星空SaaSBI是一款强大的在线商业智能工具,它通过和金蝶云星空方案的紧密合

使用docker安装配置oracle 11g

1、安装docker环境。2、开始拉取oracle镜像dockerpullregistry.cn-hangzhou.aliyuncs.com/helowin/oracle\_11g3、下载完成后,查看镜像dockerimages4、启动容器dockerrun-d-p1521:1521--nameoracle11greg

【校招VIP】专业课考点之TCP连接

考点介绍:在TCP/IP中,TCP协议通过三次握手来建立连接,从而提供可靠的连接服务。本专题主要介绍一线互联网大厂面试关于TCP连接的相关问题。专业课考点之TCP连接-相关题目及解析内容可点击文章末尾链接查看!一、考点试题1.TCP是网络传输的常用协议,下面为TCP的描述,哪项是不正确的?A.TCP提供一种面向连接的、

few shot目标检测survey paper笔记(迁移学习)

paper:Few-ShotObjectDetection:AComprehensiveSurvey(CVPR2021)metalearning需要复杂的情景训练,而迁移学习仅需在一个single-branch结构上做两步训练。常用的结构是FasterR-CNN,下面是FasterR-CNN的结构图。RPN的修改当样本

数据结构---单链表

单链表单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。链表中的数据是以结点来表示的,每个结点的构成:元素(数据元素的映象)+指针(指示后继元素存储位置),元素就是存储数据的存储单元,指针就是连接每个结点的地址数据。如图是一个结点​多个结点加上head(头结点)指针(指向了第一个结点的位置

kafka知识点汇总

kafka是什么?Kafka是一个分布式的基于发布/订阅模式的消息队列(MessageQueue),主要应用于大数据实时处理领域。Kafka是由Apache软件基金会开发的一个开源流处理平台,由Scala和Java编写。Kafka是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者在网站中的所有动作流数据。消息队列

将本地前端工程中的npm依赖上传到Nexus

【问题背景】用Nexus搭建了内网的依赖仓库,需要将前端工程中node_modules中的依赖上传到Nexus上,但是node_modules中的依赖已经是解压后的状态,如果直接机械地将其简单地打包上传到Nexus,那么无法通过npminstall下载使用。故有此文。【解决思路】前端工程中的所有npm依赖信息已经记录在

我的创作纪念日

机缘不知不觉已经过去蛮久了,为何我会到csdn中来写博客呢?最开始我只是想对于我这样的应届生如果有一个自己的博客网站对于找工作会是一个加分的项。写了20篇之后呢我就觉得我的目的不是那么功利了,因为我在学习的过程中遇到了太多的困难,都是通过别人的分享去获得解决的办法的,从别人那里去获取我便也想给别人出一份力,对于没有导师

vue前端 router路由hash和history模式区别

vue-router路由有两种方式,hash模式和history模式,接下来浅谈一下两者区别模式通俗解释兼容原理hashurl尾巴后的#号及后面的字符都是hash模式兼容IE8及以上于hash值变化不会导致浏览器向服务器发出请求,而且hash改变会触发hashchange事件(hashchange只能改变#后面的url

改写paddledetection为cmake版(c++)

下载源代码官方地址:https://gitee.com/paddlepaddle/PaddleDetection网盘:paddledetection链接:https://pan.baidu.com/s/1g0z5SYQNDR1pwe9iAtvR3A?pwd=ktl6提取码:ktl6paddleocr链接:https:/

热文推荐