今天煌sir来带你扫扫盲, 关于什么是序列化?反序列化? 场景使用?  怎么实现??? 或许我们平时大概知道, 嗯,序列化,就是实现 Serializable 接口, 那再问你,怎么实现的, 或许你就有点懵逼,这篇, 让我们来 look 一 look 吧~~~~~

目录

一.什么是序列化?反序列化?

二.  为什么要序列化和反序列化?什么场景使用?

三.  序列化有什么作用?好处?

四.如何实现 序列化和反序列化?

实现序列化的前提

实现代码

第一步:创建一个 JavaBean 对象

第二步:在测试类使用 ObjectOutputStream 对象实现序列化

第三步:在测试类使用ObjectInputStream 对象实现反序列化

问题1: 如果我们某些数据不需要序列化,怎么办?

问题2:  序列化版本问题,在完成序列化操作后,由于项目的升级或修改,可能我们会对序列化对象进行修改,比如增加某个字段,那么我们在进行反序列化就会报错:

五. 注意事项


一.什么是序列化?反序列化?

  • 序列化:指把堆内存中的 Java 对象数据,通过某种方式把对象存储到磁盘文件中或者传递给其他网络节点(在网络上传输)。这个过程称为序列化。通俗来说就是将数据结构或对象转换成二进制数据流的过程
  • 反序列化:把磁盘文件中的对象数据或者把网络节点上的对象数据,恢复成Java对象模型的过程。也就是将在序列化过程中所生成的二进制数据流转换成数据结构或者对象的过程

二.  为什么要序列化和反序列化?什么场景使用? 

  ①、在分布式系统中,此时需要把对象在网络上传输,就得把对象数据转换为二进制形式,需要共享的数据的 JavaBean 对象,都得做序列化。

  ②、服务器钝化:如果服务器发现某些对象好久没活动了,那么服务器就会把这些内存中的对象持久化在本地磁盘文件中(Java对象转换为二进制文件);如果服务器发现某些对象需要活动时,先去内存中寻找,找不到再去磁盘文件中反序列化我们的对象数据,恢复成 Java 对象。这样能节省服务器内存。

我们想一想,  当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等,这些如何实现的呢?

没错,这其中就用到了   Java序列化与反序列化了 ~~~ 这些数据都会以二进制序列的形式在网络上传送。

一句话概括就是 ,  发送方 发送数据需要把这个Java对象转换为字节序列,然后在网络上传送 ;  然后接收方需要从字节序列中恢复出Java对象。从而达到传送的目的~~

总的来说: 使用场景归纳如下:

1.把内存中的对象保存到一个文件中或者数据库中;

2. 网络上传送对象;

3.通过RMI传输对象;(RMI->Remote Method Invocation 远程方法调用)

小锦囊: RMI是什么?

RMI(Remote Method Invocation,远程方法调用)是bai用Java在JDK1.1中实现的,它du大大增强了Java开发分布式应用zhi的能力。Java作为一种风靡一dao时的网络开发语言,其巨大的威力就体现在它强大的开发分布式网络应用的能力上,而RMI就是开发百分之百纯Java的网络分布式应用系统的核心解决方案之一。其实它可以被看作是RPC的Java版本。但是传统RPC并不能很好地应用于分布式对象系统。而Java RMI 则支持存储于不同地址空间的程序级对象之间彼此进行通信,实现远程对象之间的无缝远程调用。

三.  序列化有什么作用?好处?

  • 描述数据的传输格式,这样可以方便自己组织数据传输格式,以至于避免一些麻烦及错误
  • 如果是跨平台的序列化,则发送方序列化后,接收方可以用任何其支持的平台反序列化成相应的版本,比如 Java序列化后, 用.net、phython等反序列化
  • 利用序列化实现远程通信,即在网络上传送对象的字节序列。
  • 为了解决对象流读写操作时可能引发的问题(如果不进行序列化,可能会存在数据乱序的问题)
  • 实现了数据的持久化,通过序列化可以把数据永久地保存到硬盘上(通常存放在文件里),
  • 序列化除了能够实现对象的持久化之外,还能够用于对象的深度克隆

四.如何实现 序列化和反序列化?

Java API提供了对序列化的支持

1、JDK类库中序列化和反序列化API

(1)java.io.ObjectOutputStream:表示对象输出流;

它的writeObject(Object obj)方法可以对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中;

(2)java.io.ObjectInputStream:表示对象输入流;

它的readObject()方法源输入流中读取字节序列,再把它们反序列化成为一个对象,并将其返回;

序列化图示:

反序列化图示:

  • 实现序列化的前提

必须实现了Serializable或Externalizable接口的类的对象才能被序列化,否则抛出异常!

  • 实现代码

第一步:创建一个 JavaBean 对象

package com.czxy.domain;import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;import javax.persistence.Id;
import javax.persistence.Table;
import java.io.Serializable;/*** @author 煌sir@itcast.cn* @version 1.0**/
@Table(name = "t_area")
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Area implements Serializable {@Idprivate Integer id;private String name;
}

第二步:在测试类使用 ObjectOutputStream 对象实现序列化

  @Testpublic void run1() throws Exception {Area a = new Area(1, "煌sir6666");File file = new File("E:\\file\\student.txt");  //文件位置file.createNewFile();   //自动创建文件FileOutputStream fos = new FileOutputStream(file);ObjectOutputStream oos = new ObjectOutputStream(fos);oos.writeObject(a);oos.flush();oos.close();fos.close();}

运行成功后, 在存储文件中,会有一长串乱码,这是二进制文件

当然, 我们不需要看懂, 只要电脑能识别就行了,嚯嚯~~~~

第三步:在测试类使用ObjectInputStream 对象实现反序列化

  @Testpublic void run2() throws Exception {File file = new File("E:\\file\\student.txt");  //文件位置FileInputStream fis = new FileInputStream(file);ObjectInputStream ois = new ObjectInputStream(fis);Area a = (Area) ois.readObject();System.out.println("id:" + a.getId());System.out.println("aname:" + a.getName());//关闭ois.close();fis.close();}

输出结果:

问题1: 如果我们某些数据不需要序列化,怎么办?

解决: 在字段面前加上 transient

Transient关键字的作用是控制变量的序列化,在变量声明前加上该关键字,可以阻止该变量被序列化到文件中,在被反序列化后,transient变量的值被设为初始值,如int型的是0,对象型的是 null。

transient表示瞬态的,被transient关键字修饰的变量不再能被序列化,一个静态变量不管是否被transient修饰,均不能被序列化。

 @Idprivate Integer id;  //序列化transient private String name; //不需要序列化
}

运行结果如下:

问题2:  序列化版本问题,在完成序列化操作后,由于项目的升级或修改,可能我们会对序列化对象进行修改,比如增加某个字段,那么我们在进行反序列化就会报错:

解决:

在 JavaBean 对象中增加一个 serialVersionUID 字段,用来固定这个版本,无论我们怎么修改,版本都是一致的,就能进行反序列化了

private static final long serialVersionUID = 8656128222714547171L;

五. 注意事项

1、序列化时,只对对象的状态进行保存,而不管对象的方法;

2、当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;

3、当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化;

4、并非所有的对象都可以序列化,至于为什么不可以,有很多原因了,比如:

安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行RMI传输等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的;

资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分配,而且,也是没有必要这样实现;

5、声明为static和transient类型的成员数据不能被序列化。因为static代表类的状态,transient代表对象的临时数据。

6、序列化运行时使用一个称为 serialVersionUID 的版本号与每个可序列化类相关联,该序列号在反序列化过程中用于验证序列化对象的发送者和接收者是否为该对象加载了与序列化兼容的类。为它赋予明确的值。显式地定义serialVersionUID有两种用途:

在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;

在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。

7、Java有很多基础类已经实现了serializable接口,比如String,Vector等。但是也有一些没有实现serializable接口的;

8、如果一个对象的成员变量是一个对象,那么这个对象的数据成员也会被保存!这是能用序列化解决深拷贝的重要原因;

看完恭喜你,又知道了一点点!!!

你知道的越多,不知道的越多! 

~感谢志同道合的你阅读,  你的支持是我学习的最大动力 ! 加油 ,陌生人一起努力,共勉!!

注: 如果本篇有需要改进的地方或错误,欢迎大神们指定一二~~

什么是序列化?反序列化? 场景使用? 怎么实现???相关推荐

  1. Redis Template使用append方法不起作用的解决办法以及序列化/反序列化的解释

    今天使用Redis Template进行测试发现的问题, 遇到这个错 org.springframework.data.redis.serializer.SerializationException: ...

  2. Java 序列化反序列化框架比较

    文章目录 一.简介 二.序列化框架 1.JDK 2.XML序列化 3.JSON序列化 4.Hessian 5.Avro序列化 6.Kyro序列化 7.Protostuff 三.序列化框架对比测试 1. ...

  3. 一个更好的C++序列化/反序列化库Kapok

    1.Kapok的特点 简单,易用,header-only,只需要引用Kapok.hpp即可:高效,初步测试性和messagepack相当. 它是纯c++11实现,因此需要支持C++11的编译器. 2. ...

  4. 用 XStream 序列化/反序列化 XML 为 Java 对象(实例)

    用 XStream 序列化/反序列化 XML 为 Java 对象(实例) Posted on 2007-06-05 19:06 Stephen Wong 阅读(1224) 评论(0)  编辑 收藏 网 ...

  5. 序列化反序列化api(入门级)

    定义: java序列化是指把Java对象转换为字节序列的过程:而Java反序列化是指把字节序列恢复为Java对象的过程. 为什么字符串通常也会进行序列化? 对象需要进行序列化的原因:保证对象的状态不变 ...

  6. 敏捷中gwt含义_在GWT中序列化/反序列化Json

    敏捷中gwt含义 JSON和GWT 最近, GWT用户小组中进行了有趣的讨论 ,涉及在客户端对JSON进行序列化/反序列化的最佳实践. 这篇文章旨在突出其重点. 到目前为止,在GWT中有三种将对象转换 ...

  7. 在GWT中序列化/反序列化Json

    JSON和GWT 最近, GWT用户小组中进行了有趣的讨论 ,涉及在客户端对JSON进行序列化/反序列化的最佳实践. 这篇文章旨在突出其重点. 到目前为止,在GWT中有三种将对象转换为JSON并从客户 ...

  8. 高效的序列化/反序列化数据方式 Protobuf

    高效的序列化/反序列化数据方式 Protobuf github地址 目录 protocolBuffers 序列化 Int32 String Map slice 序列化小结 protocolBuffer ...

  9. 10.8-全栈Java笔记:序列化/反序列化的步骤和实例

    本节我们详细讲解10.3节中提到的序列化和反序列化操作. 序列化和反序列化是什么 当两个进程远程通信时,彼此可以发送各种类型的数据. 无论是何种类型的数据,都会以二进制序列的形式在网络上传送.比如,我 ...

最新文章

  1. androidstudio jni开发_初识NDK开发(一)
  2. 三年后75%企业数据将移至云端?SAP新应用加速云服务
  3. python测试rabbitmq的消息收发
  4. 【干货】Duang!看好产品是如何增加情感化设计的?
  5. DIB位图(Bitmap)的读取和保存
  6. boost::geometry::comparable_distance用法的测试程序
  7. LeetCode 2. 两数相加(单链表反转)
  8. ArcGIS API for Silverlight开发入门(2):一个基础地图实例
  9. 关于批量导入数据以及调优的一些总结
  10. 识别数学公式_新品独家发布(免费送码)丨极度公式 数学公式 OCR 识别编辑 LaTeX 公式软件...
  11. MINIST手写数字数据集–神经网络(mini-batch)
  12. Mysql sql执行错误#1436 Thread stack overrun:
  13. 香港服务器防攻击能力好吗
  14. 3G中的A-GPS移动定位技术
  15. 如何查看Android API文档
  16. Android实现仿QQ登录可编辑下拉菜单
  17. 关于自建ip池可能出现的问题
  18. 一款可以对文本去重对比去重单去重查重的好用软件介绍文本综合处理软件教程
  19. proteus元件图片_proteus元件对照表(经典详细)
  20. [Linux]自动挂载fstab/挂载光盘/挂载U盘

热门文章

  1. ORACLE逐行累计求和方法
  2. 约瑟夫环问题-基础版(数学归纳法)
  3. C++实用工具类-ini配置文件解析
  4. 转:洛克菲勒写给儿子的三封家书:利益、精神、尊严
  5. 学习HTML、CSS、Javascript必看书
  6. 我是一个*** (五)
  7. 【聚能聊有奖话题】新 IoT僵尸网络IoT_reaper现身,你家的路由器、摄像头还安全吗?...
  8. 1.6 笔记本开机卡在logo页面(一直在加载转动,)关机重启几次也还是不行,自动修复也是失败
  9. 经济学本质上就是心里学
  10. vue组件间通信六种方式(完整版)