zoukankan      html  css  js  c++  java
  • java 序列化和反序列化的实现原理

    老是听说序列化反序列化,就是不知道到底什么是序列化,什么是反序列化?今天就在网上搜索学习一下,这一搜不要紧,发现自己曾经用过,竟然不知道那就是JDK类库中序列化和反序列化的API。

    ----什么是序列化?

    --1--java序列化是指把java对象转换为字节序列的过程,而java反序列化是指把字节序列恢复为java对象的过程

    --2--序列化:对象序列化的最主要的用处就是在传递和保存对象的时候,保证对象的完整性和可传递性。序列化是把对象转换成有序字节流,以便在网络上传输或者保存在本地文件中。序列化后的字节流保存的java对象的状态以及相关的描述信息。序列化机制的核心作用就是对象状态的保存与重建。

    --3--反序列化:客户端从文件中或网络上获得序列化后的对象字节流后,根据字节流中所保存的对象状态及描述信息,通过反序列化重建对象。

    --4--序列化就是把实体对象状态按照一定的格式写入到有序字节流,反序列化就是从有序字节流重建对象,恢复对象状态

    ----为什么需要序列化与反序列化

    当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本,图片,音频,视频等,而这些数据都会以二进制的形式在网络上传送。

    当两个java进行进行通信时,要传送对象,怎么传对象,通过序列化与反序列化。

    也就是说,发送方需要把对象转换为字节序列,然后在网络上传送,另一方面,接收方需要从字节序列中恢复出java对象

    ----序列化的好处

    --1--永久性保存对象,保存对象的字节序列到本地文件或者数据库中,实现了数据的持久化,通过序列化可以把数据永久的保存到硬盘上,

    --2--利用序列化实现远程通信,可以在网络上传送对象的字节序列。

    --3--在进程间传递对象

    ----序列化算法步骤

    --1--把对象实例相关的类元数据输出

    --2--递归输出类的超类描述直到不再有超类

    --3--类元数据完了以后,开始从最懂曾的超类开始输出对象实例的实际数据值。

    --4--从上至下递归输出实例的数据

     ----Java 如何实现序列化和反序列化

    --1-- JDK类库中序列化API

    java.io.ObjectOutputStream: 表示输出对象流

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

    --2--java.io.ObjectInputStream:表示对象输入流

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

    --2--实现序列化的要求

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

    --3--实现java对象序列化与反序列化的方法

    例 a 类,它的对象需要序列化,有3种方法

    如果类a仅仅实现了Serializable接口,则

    ObjectOutputStream采用默认的序列化方式,对a对象的非transient实例变量进行序列化

    ObjectInputStream采用默认的反序列化方式,对a对象的非transient实例变量进行反序列化

    如果类a仅仅实现了Serializable接口,并且还定义了a对象的writeObject(ObjectOutputStream out) 和readObject(ObjectInputStream in),则

    ObjectOutputStream调用a对象的writeObject(ObjectOutputStream out)的方法进行序列化

    ObjectInputStream调用a对象的readObject(ObjectInputStream in)的方法进行序列化

    如果a类实现了ExternaInalizable接口,且User类必须实现readExternam(ObjectInput in)和wiriteExternal(ObjectOutput out)方法,则

    ObjectOutputStream调用a对象的wiriteExternal(ObjectOutput out)的方法进行序列化

    ObjectInputStream调用a对象的readExternam(ObjectInput in)的方法进行序列化‘’

     ----JDK类库中序列化的步骤

    --1--创建一个对象输出流,它可以包装一个奇特类型的目标输出流,如文件输出流:

    objectOutputStream oos=new objectOutputStream(new FileOutStream(c:\object.out));

    --2--通过对象输出流writeObject()方法写对象:

    oos.writeObject(new a("xiaoxiao","145263","female"));

    ----JDK类库中反序列化的步骤

    --1--创建一个对象输入流,它可以包装一个其他类型输入流,如文件输入流:

    objectInputStream ois=new ObjectInputStream(new FileInputStream("object.out"));

    --2--通过对象输出流的readObject()方法读取对象:

    a aa=(a)ois.readObject();

    --3--为了正确读数据,完成反序列化,必须保证向对象输出流写对象的顺序与从对象输入流中读对象的顺序一致

     ----例子

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    import java.awt.print.Printable;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.ObjectInputStream;
    import java.io.ObjectOutputStream;
    import java.io.Serializable;
     
     
    public class SerialDemo {
           public static void main(String[] args) throws IOException,ClassNotFoundException{
                   
                  FileOutputStream fos =new FileOutputStream("object.out");
                  ObjectOutputStream oos=new ObjectOutputStream(fos);
                  User user1=new User("xiaoming","145263","female");
                  oos.writeObject(user1);
                  System.out.print(oos);
                  oos.flush();
                  oos.close();
                   
                  FileInputStream fis=new FileInputStream("object.out");
                  ObjectInputStream ois=new ObjectInputStream(fis);
                  User user2=(User) ois.readObject();
                  System.out.print(user2.getUserName("xiaoming")+" "+ user2.getPassword("145263")+" " + user2.getSex("female"));
           }
    }
     
     
     
    import java.io.Serializable;
     
    public class User implements Serializable {
           public User(String string, String string2, String string3) {
                  // TODO Auto-generated constructor stub
           }
           private String userName;
           private String password;
           private String sex;
            
           public String getUserName(String userName) {
                  return userName;
           }
           public String getPassword(String password) {
                  return password;
           }
           public String getSex(String sex) {
                  return sex;
           }
    }

     文件位置

    学习于: https://blog.csdn.net/xlgen157387/article/details/79840134

    您的资助是我最大的动力!
    金额随意,欢迎来赏!

    如果,您认为阅读这篇博客让您有些收获,不妨点击一下右下角的推荐按钮。
    如果,您希望更容易地发现我的新博客,可以关注我哦!关注我

    如果,想给予我更多的鼓励,这可以支付宝扫码哦!求打

    因为,我的写作热情也离不开您的肯定支持,感谢您的阅读,我是【未来可期】!

  • 相关阅读:
    华为交换机配置和NTP服务器同步时间
    华为交换机常用查询命令
    Linux在线添加新磁盘
    华为交换机堆叠配置
    Linux-Centos安装nginx–解压版安装
    Linux修改为国内yum源
    pipenv的使用
    使用阿里云语音合成产品,从页面获取它的voice参数列表数据
    使用【Sonatype Nexus Repository Manager】搭建内部NPM源
    Vue3.0 项目中使用事件总线
  • 原文地址:https://www.cnblogs.com/isme-zjh/p/11338161.html
Copyright © 2011-2022 走看看