java 序列化与反序列化的实例详解
1.Java序列化与反序列化
Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。
2.为什么需要序列化与反序列化
我们知道,当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等,而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,能否实现进程间的对象传送呢?答案是可以的。如何做到呢?这就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。
序列化简介
Java提供了一种对象序列化的机制,该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型。
将序列化对象写入文件之后,可以从文件中读取出来,并且对它进行反序列化,也就是说,对象的类型信息、对象的数据,还有对象中的数据类型可以用来在内存中新建对象。
整个过程都是Java虚拟机(JVM)独立的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上反序列化该对象。
类ObjectInputStream和ObjectOutputStream是高层次的数据流,它们包含序列化和反序列化对象的方法。
ObjectOutputStream类包含很多写方法来写各种数据类型,但是一个特别的方法例外:
publicfinalvoidwriteObject(Objectx)throwsIOException
上面的方法序列化一个对象,并将它发送到输出流。相似的ObjectInputStream类包含如下反序列化一个对象的方法:
publicfinalObjectreadObject()throwsIOException,ClassNotFoundException
该方法从流中取出一个对象,并将对象反序列化。它的返回值为Object,因此,你需要将它转换成合适的数据类型。
为了演示序列化在Java中是怎样工作的,我们定义Employee类,该类实现了Serializable接口。
Employee.java文件代码:
publicclassEmployeeimplementsjava.io.Serializable { publicStringname; publicStringaddress; publictransientintSSN; publicintnumber; publicvoidmailCheck() { System.out.println("Mailingacheckto"+name +""+address); } }
注意,一个类的对象要想序列化成功,必须满足两个条件:
1、该类必须实现java.io.Serializable对象。
2、该类的所有属性必须是可序列化的。如果有一个属性不是可序列化的,则该属性必须注明是短暂的。
如果你想知道一个Java标准类是否是可序列化的,请查看该类的文档。检验一个类的实例是否能序列化十分简单,只需要查看该类有没有实现java.io.Serializable接口。
序列化对象
ObjectOutputStream类用来序列化一个对象,如下的SerializeDemo例子实例化了一个Employee对象,并将该对象序列化到一个文件中。
该程序执行后,就创建了一个名为employee.ser文件。该程序没有任何输出,但是你可以通过代码研读来理解程序的作用。
注意:当序列化一个对象到文件时,按照Java的标准约定是给文件一个.ser扩展名。
SerializeDemo.java文件代码:
importjava.io.*; publicclassSerializeDemo { publicstaticvoidmain(String[]args) { Employeee=newEmployee(); e.name="ReyanAli"; e.address="PhokkaKuan,AmbehtaPeer"; e.SSN=11122333; e.number=101; try { FileOutputStreamfileOut= newFileOutputStream("employee.ser"); ObjectOutputStreamout=newObjectOutputStream(fileOut); out.writeObject(e); out.close(); fileOut.close(); System.out.println("Serializeddataissavedinemployee.ser"); }catch(IOExceptioni) { i.printStackTrace(); } } }
反序列化对象
下面的DeserializeDemo程序实现了反序列化,employee.ser存储了Employee对象。
DeserializeDemo.java文件代码:
importjava.io.*; publicclassDeserializeDemo { publicstaticvoidmain(String[]args) { Employeee=null; try { FileInputStreamfileIn=newFileInputStream("employee.ser"); ObjectInputStreamin=newObjectInputStream(fileIn); e=(Employee)in.readObject(); in.close(); fileIn.close(); }catch(IOExceptioni) { i.printStackTrace(); return; }catch(ClassNotFoundExceptionc) { System.out.println("Employeeclassnotfound"); c.printStackTrace(); return; } System.out.println("DeserializedEmployee..."); System.out.println("Name:"+e.name); System.out.println("Address:"+e.address); System.out.println("SSN:"+e.SSN); System.out.println("Number:"+e.number); } }
以上程序编译运行结果如下所示:
DeserializedEmployee... Name:ReyanAli Address:PhokkaKuan,AmbehtaPeer SSN:0 Number:101
这里要注意以下要点:
readObject()方法中的try/catch代码块尝试捕获ClassNotFoundException异常。对于JVM可以反序列化对象,它必须是能够找到字节码的类。如果JVM在反序列化对象的过程中找不到该类,则抛出一个ClassNotFoundException异常。
注意,readObject()方法的返回值被转化成Employee引用。
当对象被序列化时,属性SSN的值为111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后Employee对象的SSN属性为0。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!