zoukankan      html  css  js  c++  java
  • java对象的序列化与反序列化使用

    1.Java序列化与反序列化

         Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。

      2.为什么需要序列化与反序列化

       我们知道,当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等, 而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,能否实现进程间的对象传送呢?答案是可以的。如何做到呢?这就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。基本原理和网络通信是一致的,通过特殊的编码方式:写入数据将对象以及其内部数据编码,存在在数组或者文件里面然后发送到目的地后,在进行解码,读出数据。OK到此显示出来为我们所用即可。

       当我们明晰了为什么需要Java序列化和反序列化后,我们很自然地会想Java序列化的好处。其好处一是实现了数据的持久化,通过序列化可以把数据永久地保存到硬盘上(通常存放在文件里),二是,利用序列化实现远程通信,即在网络上传送对象的字节序列。

    3.对象序列化

      java.io.ObjectOutputStream代表对象输出流,它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。只有实现了Serializable和Externalizable接口的类的对象才能被序列化。

      java.io.ObjectInputStream代表对象输入流,它的readObject()方法从一个源输入流中读取字节序列,再把它们反序列化为一个对象,并将其返回。

     1、序列化流:把对象按照流一样的方式存入文本或者在网络中传输;  对象 ---> 流 :ObjectOutputStream
     2、反序列化流:把文本文件中的流对象数据或者网络中的流对象数据还原成对象。   流---> 对象 :ObjectInputStream
    //------------------------------------
    //第一个代码是将一个序列化对象赋值后,写到一个文件里面对应write()方法,然后再读取该文件中的数据,将该数据还原为对象read()方法。
     1 import java.io.FileInputStream;
     2 import java.io.FileNotFoundException;
     3 import java.io.FileOutputStream;
     4 import java.io.IOException;
     5 import java.io.ObjectInputStream;
     6 import java.io.ObjectOutputStream;
     7 
     8 /*
     9  * 序列化流:把对象按照流一样的方式存入文本或者在网络中传输;  对象 ---> 流 :ObjectOutputStream
    10  * 反序列化流:把文本文件中的流对象数据或者网络中的流对象数据还原成对象。   流---> 对象 :ObjectInputStream
    11  */
    12 public class ObjectStreamDemo {
    13     public static void main(String[] args) throws IOException {
    14         // 序列化数据其实就是把对象写到文本文件
    15         //write();
    16         read();
    17     }
    18 
    19     private static void read() throws IOException {
    20         // 创建反序列化流对象
    21         ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
    22                 "a.txt"));
    23         // 读取,还原对象
    24         try {
    25             Person p = (Person) ois.readObject();
    26             System.out.println(p.toString());
    27         } catch (ClassNotFoundException e) {
    28             // TODO Auto-generated catch block
    29             e.printStackTrace();
    30         }
    31         
    32         ois.close();
    33     }
    34 
    35     private static void write() throws IOException {
    36         // 创建序列化流对象
    37         // public ObjectOutputStream(OutputStream out)
    38         ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(
    39                 "a.txt"));
    40         // 创建对象
    41         Person p = new Person("java", 20);
    42         oos.writeObject(p);
    43         // 释放资源
    44         oos.close();
    45     }
    46 }

    //------------------------------对象类

    // 有的时候当我们修改了对象所属类的一点内容时,该序列化对象便不能被读取了,原因是在该对象序列化的时候系统默认给出了一个ID,一旦修改了该类该ID将会发生变化

    // 读取的时候就无法匹配,因此编译器通过黄线提示我们添加一个ID,可以使默认default的ID也可以是generated的ID都可以,我一般使用generated。

     1 import java.io.Serializable;
     2 
     3 /*
     4  * NotSerializableException为序列化异常,
     5  * 该类需要实现一个接口:Serializable序列化接口,该接口中并没有任何方法,仅仅作为标识。
     6  * 类似于此的没有方法的接口是标记接口
     7  * 
     8  * !!!每一次去修改该类的时候都会生成一个新的序列化标识的值!,需要重新新,重新读,这是基本方法。
     9  * 想办法来固定该类的标识ID,人为设定。这样即使再次修改类的内容,只要ID固定了就可以保证,在读取的时候一直是匹配的。
    10  * 增加 generated serial version ID,在类里面直接点击黄色即可,增加一个变化的ID值
    11  */
    12 
    13 /*
    14  * 当有的成员变量不需要被序列化时:如何解决。
    15  * 方法使用transient关键字声明不需要序列化的成员变量
    16  */
    17 public class Person implements Serializable{
    18 
    19     /**
    20      * serialVersionUID
    21      */
    22     private static final long serialVersionUID = -9164765814868887767L;
    23     
    24     private String name;
    25     private transient int age;
    26 
    27     public Person() {
    28         super();
    29     }
    30 
    31     public Person(String name, int age) {
    32         super();
    33         this.name = name;
    34         this.age = age;
    35     }
    36 
    37     public String getName() {
    38         return name;
    39     }
    40 
    41     public void setName(String name) {
    42         this.name = name;
    43     }
    44 
    45     public int getAge() {
    46         return age;
    47     }
    48 
    49     public void setAge(int age) {
    50         this.age = age;
    51     }
    52 
    53     @Override
    54     public String toString() {
    55         return "Person [name=" + name + ", age=" + age + "]";
    56     }
    57 
    58 }

    //上面的序列化只是简单介绍了序列化的作用,在Android开发中我们不可能为了传递一个对象重新再APK文件里面添加一个文件用来存放对象的数据,因此在Android开发室

    //直接使用的是一个字符序列数组来暂时保存序列化的二进制数值。

    //http://www.cnblogs.com/fuck1/p/5459660.html在这里面有详细的使用方法。

  • 相关阅读:
    ListComparer
    C#.net中类与结构的区别
    对C# 中堆栈,堆,值类型,引用类型的理解
    一道面试题
    Mvc不等于三层架构
    SQL养成一个好习惯是一笔财富
    清除MSSQL事务日志文件
    面向对象设计(OOD)五大基本原则
    .net/c#中栈和堆的区别及代码在栈和堆中的执行流程详解之一
    IIS7 Login failed for user IIS APPPOOL\ASP.NET v4.0
  • 原文地址:https://www.cnblogs.com/fuck1/p/5462052.html
Copyright © 2011-2022 走看看