JAVA基础(13)——序列化

JAVA基础系列规划:


本文要点:

1 序列化的必要性

分布式环境中经常需要将Object从网络或设备的一端传递到另一端,这就需要有一种可以在两端传输数据的协议,Java序列化机制即是为了解决这个问题。

2 序列化机制和API

Java序列化机制包括序列化和反序列化。序列化是将对象分解成字节流来描述的过程,以便存储在文件中或在网络中传输;反序列化是打开字节流并重建对象的过程。

Java序列化API提供一种处理对象序列化的标准机制。java.io包有两个序列化对象的类,ObjectOutputStream负责将对象写入字节流,包括对象类型和版本信息。ObjectInputStream从字节流重构对象,反序列化时JVM用头信息生成对象实例,然后将对象字节流中的数据复制到对象数据成员中。

  • Serializable和Externalizable

一个对象能够序列化的前提是实现java.io.Serializable,Serializable接口没有方法,更像是个标记。有了这个标记的Class就能被序列化机制处理。

Externalizable接口继承java.io.Serializable,当对象实现了这个接口时,通过writeExternal(ObjectOutput out)和readExternal(ObjectInput in)方法就可以灵活的控制它的序列化和反序列过程,任何实现了Externalizable接口的类都需要这实现两个函数。

  • ObjectOutputStream类和writeObject()方法

ObjectOutputStream类扩展DataOutput接口,writeObject()方法是最重要的方法,用于对象序列化。如果对象包含其他对象的引用,则writeObject()方法递归序列化这些对象。每个ObjectOutputStream维护序列化的对象引用表,防止发送同一对象的多个拷贝。

  • ObjectInputStream类和readObject()方法

ObjectInputStream类扩展DataOutput接口,readObject()方法是最重要的方法,用于从字节流中反序列化对象。readObject()收到对象时,JVM装入头中指定的类。如果找不到这个类,则readObject()抛出ClassNotFoundException,如果需要传输对象数据和字节码,则可以用RMI框架。

  • transit和static

transit变量和类变量(static)不被序列化。

3 序列化一个对象

类声明:

import java.io.Serializable;  

class TestSerial implements Serializable {  

       public byte version = 100;  

       public byte count = 0;  

} 

序列化:

public static void main(String args[]) throws IOException {  

    FileOutputStream fos = new FileOutputStream("temp.out");  
    ObjectOutputStream oos = new ObjectOutputStream(fos);  
    TestSerial ts = new TestSerial();  
    oos.writeObject(ts);  
    oos.flush();  
    oos.close();  
} 

执行结果为,生成temp.out文件,以16进制方式显示。内容应该如下:

AC ED 00 05 73 72 00 0A 53 65 72 69 61 6C 54 65
73 74 A0 0C 34 00 FE B1 DD F9 02 00 02 42 00 05
63 6F 75 6E 74 42 00 07 76 65 72 73 69 6F 6E 78
70 00 64

我们注意到TestSerial类中只有两个域:

public byte version = 100;
public byte count = 0;

且都是byte型,理论上存储这两个域只需要2个byte,但是实际上temp.out占据空间为51bytes,也就是说除了数据以外,还包括了对序列化对象的其他描述。

反序列化:

public static void main(String args[]) throws IOException {  

    FileInputStream fis = new FileInputStream("temp.out");  
    ObjectInputStream oin = new ObjectInputStream(fis);  
    TestSerial ts = (TestSerial) oin.readObject();  
    System.out.println("version="+ts.version);  
} 

执行结果为

version=100

4 Java序列化算法过程

序列化算法一般会按步骤做如下事情:

  • 将对象实例相关的类元数据输出;
  • 递归地输出类的超类描述直到不再有超类;
  • 类元数据完了以后,开始从最顶层的超类开始输出对象实例的实际数据值;
  • 从上至下递归输出实例的数据。

用另一个更完整覆盖所有可能出现的情况的例子来说明:

class parent implements Serializable {  
       int parentVersion = 10;  
}  

class contain implements Serializable{  
       int containVersion = 11;  
}  

public class SerialTest extends parent implements Serializable {  

       int version = 66;  
       contain con = new contain();  

       public int getVersion() {  
              return version;  
       }  

       public static void main(String args[]) throws IOException {  
              FileOutputStream fos = new FileOutputStream("temp.out");  
              ObjectOutputStream oos = new ObjectOutputStream(fos);  
              SerialTest st = new SerialTest();  
              oos.writeObject(st);  
              oos.flush();  
              oos.close();  
       }  
} 

SerialTest类实现了Parent超类,内部还持有一个Container对象。序列化后的格式如下:

AC ED 00 05 73 72 00 0A 53 65 72 69 61 6C 54 65
73 74 05 52 81 5A AC 66 02 F6 02 00 02 49 00 07
76 65 72 73 69 6F 6E 4C 00 03 63 6F 6E 74 00 09
4C 63 6F 6E 74 61 69 6E 3B 78 72 00 06 70 61 72
65 6E 74 0E DB D2 BD 85 EE 63 7A 02 00 01 49 00
0D 70 61 72 65 6E 74 56 65 72 73 69 6F 6E 78 70
00 00 00 0A 00 00 00 42 73 72 00 07 63 6F 6E 74
61 69 6E FC BB E6 0E FB CB 60 C7 02 00 01 49 00
0E 63 6F 6E 74 61 69 6E 56 65 72 73 69 6F 6E 78
70 00 00 00 0B

下面来仔细看看这些字节的含义。

AC ED 00 05 73
  • AC ED: STREAM_MAGIC. 声明使用了序列化协议.
  • 00 05: STREAM_VERSION. 序列化协议版本.
  • 73: TC_OBJECT. 声明这是一个新的对象.

序列化算法的第一步就是输出对象相关类的描述。例子所示对象为SerialTest类实例,因此接下来输出SerialTest类的描述:

72 00 0A 53 65 72 69 61 6C 54 65 73 74 05 52 81 5A AC 66 02 F6 02 00 02
  • 72: TC_CLASSDESC. 声明这里开始一个新Class。
  • 00 0A: Class名字的长度.
  • 53 65 72 69 61 6c 54 65 73 74: SerialTest,Class类名.
  • 05 52 81 5A AC 66 02 F6: SerialVersionUID, 序列化ID,如果没有指定,则会由算法随机生成一个8byte的ID.
  • 02: 标记号. 该值声明该对象支持序列化。
  • 00 02: 该类所包含的域个数。

接下来,算法输出其中的一个域,int version=66;

49 00 07 76 65 72 73 69 6F 6E
  • 49: 域类型. 49 代表”I”, 也就是Int.
  • 00 07: 域名字的长度.
  • 76 65 72 73 69 6F 6E: version,域名字描述.

然后,算法输出下一个域,contain con = new contain();这个有点特殊,是个对象。描述对象类型引用时需要使用JVM的标准对象签名表示法:

4C 00 03 63 6F 6E 74 00 09 4C 63 6F 6E 74 61 69 6E 3B 78
  • 4C: 域的类型.
  • 00 03: 域名字长度.
  • 63 6F 6E: 域名字描述,con
  • 74: TC_STRING. 代表一个new String.用String来引用对象。
  • 00 09: 该String长度.
  • 4C 63 6F 6E 74 61 69 6E 3B: Lcontain;, JVM的标准对象签名表示法.
  • 78: TC_ENDBLOCKDATA,对象数据块结束的标志.

接下来算法就会输出超类也就是Parent类描述了:

72 00 06 70 61 72 65 6E 74 0E DB D2 BD 85 EE 63 7A 02 00 01 
  • 72: TC_CLASSDESC. 声明这个是个新类.
  • 00 06: 类名长度.
  • 70 61 72 65 6E 74: parent,类名描述。
  • 0E DB D2 BD 85 EE 63 7A: SerialVersionUID, 序列化ID.
  • 02: 标记号. 该值声明该对象支持序列化.
  • 00 01: 类中域的个数.

下一步,输出parent类的域描述,int parentVersion=100;

49 00 0D 70 61 72 65 6E 74 56 65 72 73 69 6F 6E 78 70
  • 49: 域类型. 49 代表”I”, 也就是Int.
  • 00 0D: 域名字长度.
  • 70 61 72 65 6E 74 56 65 72 73 69 6F 6E: parentVersion,域名字描述。
  • 78: TC_ENDBLOCKDATA,对象块结束的标志。
  • 70: TC_NULL, 说明没有其他超类的标志。

到此为止,算法已经对所有的类的描述都做了输出。下一步就是把实例对象的实际值输出了。这时候是从parent Class的域开始的:

00 00 00 0A 00 00 00 42
  • 00 00 00 0A: 10, parentVersion域的值.
  • 00 00 00 42: 66, SerialTest类的域version的值.

再往后的bytes比较有意思,算法需要描述contain类的信息,要记住,现在还没有对contain类进行过描述:

73 72 00 07 63 6F 6E 74 61 69 6E FC BB E6 0E FB CB 60 C7 02 00 01 
  • 73: TC_OBJECT, 声明这是一个新的对象.
  • 72: TC_CLASSDESC声明这里开始一个新Class.
  • 00 07: 类名的长度.
  • 63 6F 6E 74 61 69 6E: contain,类名描述.
  • FC BB E6 0E FB CB 60 C7: SerialVersionUID, 序列化ID.
  • 02: Various flags. 标记号. 该值声明该对象支持序列化
  • 00 01: 类内的域个数。

输出contain的唯一的域描述,int containVersion=11;

49 00 0E 63 6F 6E 74 61 69 6E 56 65 72 73 69 6F 6E 78 
  • 49: 域类型. 49 代表”I”, 也就是Int..
  • 00 0E: 域名字长度.
  • 63 6F 6E 74 61 69 6E 56 65 72 73 69 6F 6E: containVersion, 域名字描述.
  • 78: TC_ENDBLOCKDATA对象块结束的标志.

这时,序列化算法会检查contain是否有超类,如果有的话会接着输出。

70 
  • 70:TC_NULL,没有超类了。

最后,将contain类实际域值输出:

00 00 00 0B
  • 00 00 00 0B: 11, containVersion的值.

5 注意事项

  • 指定序列版本UID;

参考文档

Java序列化的机制和原理
Java 序列化的高级认识
Java基础学习总结——Java对象的序列化和反序列化

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值