java的java反序列化化swfit能不能解析

Java基础(42)
&java序列化和反序列化
& & & &java对象序列化的意思就是将对象的状态转化成字节流,以后可以通过这些值再生成相同状态的对象。对象序列化是对象持久化的一种实现方法,它是将对象的属性和方法转化为一种序列化的形式用于存储和传输。反序列化就是根据这些保存的信息重建对象的过程。
& & & &序列化:将java对象转化为字节序列的过程。
& & & &反序列化:将字节序列转化为java对象的过程。
二、为什么要序列化和反序列化
& &&& &我们知道,当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等,&而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,能否实现进程间的对象传送呢?答案是可以的。如何做到呢?这就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。当我们明晰了为什么需要Java序列化和反序列化后,我们很自然地会想Java序列化的好处。其好处一是实现了数据的持久化,通过序列化可以把数据永久地保存到硬盘上(通常存放在文件里),二是,利用序列化实现远程通信,即在网络上传送对象的字节序列。
三、涉及到的javaAPI 
& & & & &&它的writeObject(Object
obj)方法可以对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。
& & & & &&它的readObject()方法源输入流中读取字节序列,再把它们反序列化成为一个对象,并将其返回。
& & & & &只有实现了Serializable或Externalizable接口的类的对象才能被序列化,否则抛出异常。
四、序列化和反序列化的步骤
& & & & &序列化:
& & & & & &步骤一:创建一个对象输出流,它可以包装一个其它类型的目标输出流,如文件输出流:
& & & & &步骤二:通过对象输出流的方法写对象:
步骤一:创建一个对象输入流,它可以包装一个其它类型输入流,如文件输入流:
& & & & &步骤二:通过对象输出流的方法读取对象:
& & & & & & & & & & & & String obj1 = (String)in.readObject();
& & & & & & & & & & & & Date obj2 = &(Date)in.readObject();
& & & & 说明:为了正确读取数据,完成反序列化,必须保证向对象输出流写对象的顺序与从对象输入流中读对象的顺序一致。
五、举个例子
& & && 我们首先写个Person实现Serializable接口:
& & & & &其次,我们在main()里面写个方法,执行序列化过程:
& & & && &代码很简单,首先创建一个对象输出流ObjectOutputStream,它可以包装一个其它类型的目标输出流,如文件输出流FileOutputStream,并指定存储的位置为“E:/hello.txt”,然后通过对象输出流的方法写对象便执行了序列化过程。运行看一下效果,正确的话便会在控制台打印“”,并且在本地E盘下会创建一个Hello.txt文件。
& & & & 我们查看一下hello.txt文件中的内容,里面是一串字节序列,打开该文件的时候不要用自带的记事本打开,因为涉及到字符编码的问题,所以显示的话是一串乱码,建议用SublimeText打开。
& & & & 我们再写个方法来反序列化该字节成Person对象,并打印出里面的值。
& & & & 执行反序列化的代码也是很简单的,首先创建一个输入流对象ObjectInputStream,然后从指定的目录下“E:/hello.txt”获取它的字节序列,然后通过输入流对象的readObject()方法将其获得的对象强制转化为Person对象,这就完成了反序列化工作,正确的反序列化成功的情况下控制台打印输出为:
&序列化ID的作用
& & & &有关序列化和反序列化的概念,可以查看前一篇的讲解,这一篇主要说明一下序列化过程中出现的问题即java序列化和反序列化中ID的作用。
& & & & 在前一篇的介绍中,我们在代码里会发现有这样一个变量:serialVersionUID,那么这个变量serialVersionUID到底具有什么作用呢?能不能去掉呢?
& & & &序列化ID的作用:&&
& & & &其实,这个序列化ID起着关键的作用,它决定着是否能够成功反序列化!简单来说,java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的。在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地实体类中的serialVersionUID进行比较,如果相同则认为是一致的,便可以进行反序列化,否则就会报序列化版本不一致的异常。等会我们可以通过代码验证一下。
& & & &序列化ID如何产生:
& & & &当我们一个实体类中没有显示的定义一个名为“serialVersionUID”、类型为long的变量时,Java序列化机制会根据编译时的class自动生成一个serialVersionUID作为序列化版本比较,这种情况下,只有同一次编译生成的class才会生成相同的serialVersionUID。譬如,当我们编写一个类时,随着时间的推移,我们因为需求改动,需要在本地类中添加其他的字段,这个时候再反序列化时便会出现serialVersionUID不一致,导致反序列化失败。那么如何解决呢?便是在本地类中添加一个“serialVersionUID”变量,值保持不变,便可以进行序列化和反序列化。
& & & &验证“serialVersionUID”不一致导致反序列化失败
& & & & 复用前篇使用到的代码,首先,我们生成一个本地Person类,里面添加一个字段age,然后将其序列化存于本地E:/hello.txt中,
& & & & 运行一下,会在控制台中打印“序列化成功。”,然后我们在Person类中再添加一个字段,name,然后直接从E:/hello.txt中反序列化,再运行一下,看看会出现什么问题。
& & & &运行一下,不出意外,报了一个异常。
& & & 从上面两张图便可以看出两次的序列化ID是不一样的,导致反序列化失败。
& & &虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致(就是 private static final long serialVersionUID = 1L)。
&&相关文章推荐
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:49271次
积分:3586
积分:3586
排名:第9221名
原创:314篇
转载:55篇
评论:14条
文章:16篇
阅读:3352
文章:11篇
阅读:2170
文章:13篇
阅读:4571
文章:15篇
阅读:3268中国领先的IT技术网站
51CTO旗下网站
深入理解Java对象序列化
关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结。此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制。
作者:Sha Jiang来源:Sha Jiang的博客| 10:29
关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结。此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制。在撰写本文时,既参考了Thinking in Java, Effective Java,JavaWorld,developerWorks中的相关文章和其它网络资料,也加入了自己的实践经验与理解,文、码并茂,希望对大家有所帮助。(持续更新中,最后更新)
1. 什么是Java对象序列化
Java平台允许我们在内存中创建可复用的Java对象,但一般情况下,只有当JVM处于运行时,这些对象才可能存在,即,这些对象的生命周期不会比JVM的生命周期更长。但在现实应用中,就可能要求在JVM停止运行之后能够保存(持久化)指定的对象,并在将来重新读取被保存的对象。Java对象序列化就能够帮助我们实现该功能。
使用Java对象序列化,在保存对象时,会把其状态保存为一组字节,在未来,再将这些字节组装成对象。必须注意地是,对象序列化保存的是对象的&状态&,即它的成员变量。由此可知,对象序列化不会关注类中的静态变量。
除了在持久化对象时会用到对象序列化之外,当使用RMI(远程方法调用),或在网络中传递对象时,都会用到对象序列化。Java序列化API为处理对象序列化提供了一个标准机制,该API简单易用,在本文的后续章节中将会陆续讲到。
2. 简单示例
在Java中,只要一个类实现了java.io.Serializable接口,那么它就可以被序列化。此处将创建一个可序列化的类Person,本文中的所有示例将围绕着该类或其修改版。
Gender类,是一个枚举类型,表示性别
public&enum&Gender&{ &&&&&MALE,&FEMALE &}&
如果熟悉Java枚举类型的话,应该知道每个枚举类型都会默认继承类java.lang.Enum,而该类实现了Serializable接口,所以枚举类型对象都是默认可以被序列化的。
Person类,实现了Serializable接口,它包含三个字段:name,String类型;age,Integer类型;gender,Gender类型。另外,还重写该类的toString()方法,以方便打印Person实例中的内容。
public&class&Person&implements&Serializable&{ &&&&&&private&String&name&=&null; &&&&&&private&Integer&age&=&null; &&&&&&private&Gender&gender&=&null; &&&&&&public&Person()&{ &&&&&&&&&System.out.println(&none-arg&constructor&); &&&&&} &&&&&&public&Person(String&name,&Integer&age,&Gender&gender)&{ &&&&&&&&&System.out.println(&arg&constructor&); &&&&&&&&&this.name&=& &&&&&&&&&this.age&=& &&&&&&&&&this.gender&=& &&&&&} &&&&&&public&String&getName()&{ &&&&&&&&&return& &&&&&} &&&&&&public&void&setName(String&name)&{ &&&&&&&&&this.name&=& &&&&&} &&&&&&public&Integer&getAge()&{ &&&&&&&&&return& &&&&&} &&&&&&public&void&setAge(Integer&age)&{ &&&&&&&&&this.age&=& &&&&&} &&&&&&public&Gender&getGender()&{ &&&&&&&&&return& &&&&&} &&&&&&public&void&setGender(Gender&gender)&{ &&&&&&&&&this.gender&=& &&&&&} &&&&&&@Override&&&&&public&String&toString()&{ &&&&&&&&&return&&[&&+&name&+&&,&&&+&age&+&&,&&&+&gender&+&&]&; &&&&&} &}&
SimpleSerial,是一个简单的序列化程序,它先将一个Person对象保存到文件person.out中,然后再从该文件中读出被存储的Person对象,并打印该对象。
public&class&SimpleSerial&{ &&&&&&public&static&void&main(String[]&args)&throws&Exception&{ &&&&&&&&&File&file&=&new&File(&person.out&); &&&&&&&&&&ObjectOutputStream&oout&=&new&ObjectOutputStream(new&FileOutputStream(file)); &&&&&&&&&Person&person&=&new&Person(&John&,&101,&Gender.MALE); &&&&&&&&&oout.writeObject(person); &&&&&&&&&oout.close(); &&&&&&&&&&ObjectInputStream&oin&=&new&ObjectInputStream(new&FileInputStream(file)); &&&&&&&&&Object&newPerson&=&oin.readObject();&&&&&&&&&&oin.close(); &&&&&&&&&System.out.println(newPerson); &&&&&} &}&
上述程序的输出的结果为:
arg&constructor &[John,&31,&MALE]&
此时必须注意的是,当重新读取被保存的Person对象时,并没有调用Person的任何构造器,看起来就像是直接使用字节将Person对象还原出来的。
当Person对象被保存到person.out文件中之后,我们可以在其它地方去读取该文件以还原对象,但必须确保该读取程序的CLASSPATH中包含有Person.class(哪怕在读取Person对象时并没有显示地使用Person类,如上例所示),否则会抛出ClassNotFoundException。
3. Serializable的作用
为什么一个类实现了Serializable接口,它就可以被序列化呢?在上节的示例中,使用ObjectOutputStream来持久化对象,在该类中有如下代码:
private&void&writeObject0(Object&obj,&boolean&unshared)&throws&IOException&{ &&&&& &...&&&&if&(obj&instanceof&String)&{ &&&&&&&&&writeString((String)&obj,&unshared); &&&&&}&else&if&(cl.isArray())&{ &&&&&&&&&writeArray(obj,&desc,&unshared); &&&&&}&else&if&(obj&instanceof&Enum)&{ &&&&&&&&&writeEnum((Enum)&obj,&desc,&unshared); &&&&&}&else&if&(obj&instanceof&Serializable)&{ &&&&&&&&&writeOrdinaryObject(obj,&desc,&unshared); &&&&&}&else&{ &&&&&&&&&if&(extendedDebugInfo)&{ &&&&&&&&&&&&&throw&new&NotSerializableException(cl.getName()&+&&\n&&&&&&&&&&&&&&&&&&&&&&+&debugInfoStack.toString()); &&&&&&&&&}&else&{ &&&&&&&&&&&&&throw&new&NotSerializableException(cl.getName()); &&&&&&&&&} &&&&&} &&&&&... &}&
从上述代码可知,如果被写对象的类型是String,或数组,或Enum,或Serializable,那么就可以对该对象进行序列化,否则将抛出NotSerializableException。
4. 默认序列化机制
如果仅仅只是让某个类实现Serializable接口,而没有其它任何处理的话,则就是使用默认序列化机制。使用默认机制,在序列化对象时,不仅会序列化当前对象本身,还会对该对象引用的其它对象也进行序列化,同样地,这些其它对象引用的另外对象也将被序列化,以此类推。所以,如果一个对象包含的成员变量是容器类对象,而这些容器所含有的元素也是容器类对象,那么这个序列化的过程就会较复杂,开销也较大。
5. 影响序列化
在现实应用中,有些时候不能使用默认序列化机制。比如,希望在序列化过程中忽略掉敏感数据,或者简化序列化过程。下面将介绍若干影响序列化的方法。
5.1 transient关键字
当某个字段被声明为transient后,默认序列化机制就会忽略该字段。此处将Person类中的age字段声明为transient,如下所示,
public&class&Person&implements&Serializable&{ &&&&&... &&&&&transient&private&Integer&age&=&null; &&&&&... &}&
再执行SimpleSerial应用程序,会有如下输出:
arg&constructor &[John,&null,&MALE]&
可见,age字段未被序列化。
5.2 writeObject()方法与readObject()方法
对于上述已被声明为transitive的字段age,除了将transitive关键字去掉之外,是否还有其它方法能使它再次可被序列化?方法之一就是在Person类中添加两个方法:writeObject()与readObject(),如下所示:
public&class&Person&implements&Serializable&{ &&&&&... &&&&&transient&private&Integer&age&=&null; &&&&&... &&&&&&private&void&writeObject(ObjectOutputStream&out)&throws&IOException&{ &&&&&&&&&out.defaultWriteObject(); &&&&&&&&&out.writeInt(age); &&&&&} &&&&&&private&void&readObject(ObjectInputStream&in)&throws&IOException,&ClassNotFoundException&{ &&&&&&&&&in.defaultReadObject(); &&&&&&&&&age&=&in.readInt(); &&&&&} &}&
在writeObject()方法中会先调用ObjectOutputStream中的defaultWriteObject()方法,该方法会执行默认的序列化机制,如5.1节所述,此时会忽略掉age字段。然后再调用writeInt()方法显示地将age字段写入到ObjectOutputStream中。readObject()的作用则是针对对象的读取,其原理与writeObject()方法相同。再次执行SimpleSerial应用程序,则又会有如下输出:
arg&constructor &[John,&31,&MALE]&
必须注意地是,writeObject()与readObject()都是private方法,那么它们是如何被调用的呢?毫无疑问,是使用反射。详情可以看看ObjectOutputStream中的writeSerialData方法,以及ObjectInputStream中的readSerialData方法。
5.3 Externalizable接口
无论是使用transient关键字,还是使用writeObject()和readObject()方法,其实都是基于Serializable接口的序列化。JDK中提供了另一个序列化接口--Externalizable,使用该接口之后,之前基于Serializable接口的序列化机制就将失效。此时将Person类作如下修改,
public&class&Person&implements&Externalizable&{ &&&&&&private&String&name&=&null; &&&&&&transient&private&Integer&age&=&null; &&&&&&private&Gender&gender&=&null; &&&&&&public&Person()&{ &&&&&&&&&System.out.println(&none-arg&constructor&); &&&&&} &&&&&&public&Person(String&name,&Integer&age,&Gender&gender)&{ &&&&&&&&&System.out.println(&arg&constructor&); &&&&&&&&&this.name&=& &&&&&&&&&this.age&=& &&&&&&&&&this.gender&=& &&&&&} &&&&&&private&void&writeObject(ObjectOutputStream&out)&throws&IOException&{ &&&&&&&&&out.defaultWriteObject(); &&&&&&&&&out.writeInt(age); &&&&&} &&&&&&private&void&readObject(ObjectInputStream&in)&throws&IOException,&ClassNotFoundException&{ &&&&&&&&&in.defaultReadObject(); &&&&&&&&&age&=&in.readInt(); &&&&&} &&&&&&@Override&&&&&public&void&writeExternal(ObjectOutput&out)&throws&IOException&{ &&&&&&} &&&&&&@Override&&&&&public&void&readExternal(ObjectInput&in)&throws&IOException,&ClassNotFoundException&{ &&&&&&} &&&&&... &}&
此时再执行SimpleSerial程序之后会得到如下结果:
arg&constructor &none-arg&constructor &[null,&null,&null]&
从该结果,一方面,可以看出Person对象中任何一个字段都没有被序列化。另一方面,如果细心的话,还可以发现这此次序列化过程调用了Person类的无参构造器。
Externalizable继承于Serializable,当使用该接口时,序列化的细节需要由程序员去完成。如上所示的代码,由于writeExternal()与readExternal()方法未作任何处理,那么该序列化行为将不会保存/读取任何一个字段。这也就是为什么输出结果中所有字段的值均为空。
另外,使用Externalizable进行序列化时,当读取对象时,会调用被序列化类的无参构造器去创建一个新的对象,然后再将被保存对象的字段的值分别填充到新对象中。这就是为什么在此次序列化过程中Person类的无参构造器会被调用。由于这个原因,实现Externalizable接口的类必须要提供一个无参的构造器,且它的访问权限为public。
对上述Person类进行进一步的修改,使其能够对name与age字段进行序列化,但忽略掉gender字段,如下代码所示:
public&class&Person&implements&Externalizable&{ &&&&&&private&String&name&=&null; &&&&&&transient&private&Integer&age&=&null; &&&&&&private&Gender&gender&=&null; &&&&&&public&Person()&{ &&&&&&&&&System.out.println(&none-arg&constructor&); &&&&&} &&&&&&public&Person(String&name,&Integer&age,&Gender&gender)&{ &&&&&&&&&System.out.println(&arg&constructor&); &&&&&&&&&this.name&=& &&&&&&&&&this.age&=& &&&&&&&&&this.gender&=& &&&&&} &&&&&&private&void&writeObject(ObjectOutputStream&out)&throws&IOException&{ &&&&&&&&&out.defaultWriteObject(); &&&&&&&&&out.writeInt(age); &&&&&} &&&&&&private&void&readObject(ObjectInputStream&in)&throws&IOException,&ClassNotFoundException&{ &&&&&&&&&in.defaultReadObject(); &&&&&&&&&age&=&in.readInt(); &&&&&} &&&&&&@Override&&&&&public&void&writeExternal(ObjectOutput&out)&throws&IOException&{ &&&&&&&&&out.writeObject(name); &&&&&&&&&out.writeInt(age); &&&&&} &&&&&&@Override&&&&&public&void&readExternal(ObjectInput&in)&throws&IOException,&ClassNotFoundException&{ &&&&&&&&&name&=&(String)&in.readObject(); &&&&&&&&&age&=&in.readInt(); &&&&&} &&&&&... &}&
执行SimpleSerial之后会有如下结果:
arg&constructor &none-arg&constructor &[John,&31,&null]&
5.4 readResolve()方法
当我们使用Singleton模式时,应该是期望某个类的实例应该是唯一的,但如果该类是可序列化的,那么情况可能略有不同。此时对第2节使用的Person类进行修改,使其实现Singleton模式,如下所示:
public&class&Person&implements&Serializable&{ &&&&&&private&static&class&InstanceHolder&{ &&&&&&&&&private&static&final&Person&instatnce&=&new&Person(&John&,&31,&Gender.MALE); &&&&&} &&&&&&public&static&Person&getInstance()&{ &&&&&&&&&return&InstanceHolder. &&&&&} &&&&&&private&String&name&=&null; &&&&&&private&Integer&age&=&null; &&&&&&private&Gender&gender&=&null; &&&&&&private&Person()&{ &&&&&&&&&System.out.println(&none-arg&constructor&); &&&&&} &&&&&&private&Person(String&name,&Integer&age,&Gender&gender)&{ &&&&&&&&&System.out.println(&arg&constructor&); &&&&&&&&&this.name&=& &&&&&&&&&this.age&=& &&&&&&&&&this.gender&=& &&&&&} &&&&&... &}&
同时要修改SimpleSerial应用,使得能够保存/获取上述单例对象,并进行对象相等性比较,如下代码所示:
public&class&SimpleSerial&{ &&&&&&public&static&void&main(String[]&args)&throws&Exception&{ &&&&&&&&&File&file&=&new&File(&person.out&); &&&&&&&&&ObjectOutputStream&oout&=&new&ObjectOutputStream(new&FileOutputStream(file)); &&&&&&&&&oout.writeObject(Person.getInstance());&&&&&&&&&&oout.close(); &&&&&&&&&&ObjectInputStream&oin&=&new&ObjectInputStream(new&FileInputStream(file)); &&&&&&&&&Object&newPerson&=&oin.readObject(); &&&&&&&&&oin.close(); &&&&&&&&&System.out.println(newPerson); &&&&&&&&&&System.out.println(Person.getInstance()&==&newPerson);&&&&&&} &}&
执行上述应用程序后会得到如下结果:
arg&constructor &[John,&31,&MALE] &false&
值得注意的是,从文件person.out中获取的Person对象与Person类中的单例对象并不相等。为了能在序列化过程仍能保持单例的特性,可以在Person类中添加一个readResolve()方法,在该方法中直接返回Person的单例对象,如下所示:
public&class&Person&implements&Serializable&{ &&&&&&private&static&class&InstanceHolder&{ &&&&&&&&&private&static&final&Person&instatnce&=&new&Person(&John&,&31,&Gender.MALE); &&&&&} &&&&&&public&static&Person&getInstance()&{ &&&&&&&&&return&InstanceHolder. &&&&&} &&&&&&private&String&name&=&null; &&&&&&private&Integer&age&=&null; &&&&&&private&Gender&gender&=&null; &&&&&&private&Person()&{ &&&&&&&&&System.out.println(&none-arg&constructor&); &&&&&} &&&&&&private&Person(String&name,&Integer&age,&Gender&gender)&{ &&&&&&&&&System.out.println(&arg&constructor&); &&&&&&&&&this.name&=& &&&&&&&&&this.age&=& &&&&&&&&&this.gender&=& &&&&&} &&&&&&private&Object&readResolve()&throws&ObjectStreamException&{ &&&&&&&&&return&InstanceHolder. &&&&&} &&&&&... &}&
再次执行本节的SimpleSerial应用后将如下输出:
arg&constructor &[John,&31,&MALE] &true&
无论是实现Serializable接口,或是Externalizable接口,当从I/O流中读取对象时,readResolve()方法都会被调用到。实际上就是用readResolve()中返回的对象直接替换在反序列化过程中创建的对象。
原文链接:
【编辑推荐】
【责任编辑: TEL:(010)】
大家都在看猜你喜欢
头条头条头条头条头条
24H热文一周话题本月最赞
讲师:3人学习过
讲师:2人学习过
讲师:6人学习过
精选博文论坛热帖下载排行
本书论述了软件开发价值增加的思维方式。这一思维方式构成了VSTS的基础,包括VSTS的指导思想,为什么这些指导思想会以某些方式表现,以及它...
订阅51CTO邮刊Java基础(42)
&java序列化和反序列化
& & & &java对象序列化的意思就是将对象的状态转化成字节流,以后可以通过这些值再生成相同状态的对象。对象序列化是对象持久化的一种实现方法,它是将对象的属性和方法转化为一种序列化的形式用于存储和传输。反序列化就是根据这些保存的信息重建对象的过程。
& & & &序列化:将java对象转化为字节序列的过程。
& & & &反序列化:将字节序列转化为java对象的过程。
二、为什么要序列化和反序列化
& &&& &我们知道,当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等,&而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,能否实现进程间的对象传送呢?答案是可以的。如何做到呢?这就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。当我们明晰了为什么需要Java序列化和反序列化后,我们很自然地会想Java序列化的好处。其好处一是实现了数据的持久化,通过序列化可以把数据永久地保存到硬盘上(通常存放在文件里),二是,利用序列化实现远程通信,即在网络上传送对象的字节序列。
三、涉及到的javaAPI 
& & & & &&它的writeObject(Object
obj)方法可以对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。
& & & & &&它的readObject()方法源输入流中读取字节序列,再把它们反序列化成为一个对象,并将其返回。
& & & & &只有实现了Serializable或Externalizable接口的类的对象才能被序列化,否则抛出异常。
四、序列化和反序列化的步骤
& & & & &序列化:
& & & & & &步骤一:创建一个对象输出流,它可以包装一个其它类型的目标输出流,如文件输出流:
& & & & &步骤二:通过对象输出流的方法写对象:
步骤一:创建一个对象输入流,它可以包装一个其它类型输入流,如文件输入流:
& & & & &步骤二:通过对象输出流的方法读取对象:
& & & & & & & & & & & & String obj1 = (String)in.readObject();
& & & & & & & & & & & & Date obj2 = &(Date)in.readObject();
& & & & 说明:为了正确读取数据,完成反序列化,必须保证向对象输出流写对象的顺序与从对象输入流中读对象的顺序一致。
五、举个例子
& & && 我们首先写个Person实现Serializable接口:
& & & & &其次,我们在main()里面写个方法,执行序列化过程:
& & & && &代码很简单,首先创建一个对象输出流ObjectOutputStream,它可以包装一个其它类型的目标输出流,如文件输出流FileOutputStream,并指定存储的位置为“E:/hello.txt”,然后通过对象输出流的方法写对象便执行了序列化过程。运行看一下效果,正确的话便会在控制台打印“”,并且在本地E盘下会创建一个Hello.txt文件。
& & & & 我们查看一下hello.txt文件中的内容,里面是一串字节序列,打开该文件的时候不要用自带的记事本打开,因为涉及到字符编码的问题,所以显示的话是一串乱码,建议用SublimeText打开。
& & & & 我们再写个方法来反序列化该字节成Person对象,并打印出里面的值。
& & & & 执行反序列化的代码也是很简单的,首先创建一个输入流对象ObjectInputStream,然后从指定的目录下“E:/hello.txt”获取它的字节序列,然后通过输入流对象的readObject()方法将其获得的对象强制转化为Person对象,这就完成了反序列化工作,正确的反序列化成功的情况下控制台打印输出为:
&序列化ID的作用
& & & &有关序列化和反序列化的概念,可以查看前一篇的讲解,这一篇主要说明一下序列化过程中出现的问题即java序列化和反序列化中ID的作用。
& & & & 在前一篇的介绍中,我们在代码里会发现有这样一个变量:serialVersionUID,那么这个变量serialVersionUID到底具有什么作用呢?能不能去掉呢?
& & & &序列化ID的作用:&&
& & & &其实,这个序列化ID起着关键的作用,它决定着是否能够成功反序列化!简单来说,java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的。在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地实体类中的serialVersionUID进行比较,如果相同则认为是一致的,便可以进行反序列化,否则就会报序列化版本不一致的异常。等会我们可以通过代码验证一下。
& & & &序列化ID如何产生:
& & & &当我们一个实体类中没有显示的定义一个名为“serialVersionUID”、类型为long的变量时,Java序列化机制会根据编译时的class自动生成一个serialVersionUID作为序列化版本比较,这种情况下,只有同一次编译生成的class才会生成相同的serialVersionUID。譬如,当我们编写一个类时,随着时间的推移,我们因为需求改动,需要在本地类中添加其他的字段,这个时候再反序列化时便会出现serialVersionUID不一致,导致反序列化失败。那么如何解决呢?便是在本地类中添加一个“serialVersionUID”变量,值保持不变,便可以进行序列化和反序列化。
& & & &验证“serialVersionUID”不一致导致反序列化失败
& & & & 复用前篇使用到的代码,首先,我们生成一个本地Person类,里面添加一个字段age,然后将其序列化存于本地E:/hello.txt中,
& & & & 运行一下,会在控制台中打印“序列化成功。”,然后我们在Person类中再添加一个字段,name,然后直接从E:/hello.txt中反序列化,再运行一下,看看会出现什么问题。
& & & &运行一下,不出意外,报了一个异常。
& & & 从上面两张图便可以看出两次的序列化ID是不一样的,导致反序列化失败。
& & &虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致(就是 private static final long serialVersionUID = 1L)。
&&相关文章推荐
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:49273次
积分:3586
积分:3586
排名:第9221名
原创:314篇
转载:55篇
评论:14条
文章:16篇
阅读:3352
文章:11篇
阅读:2170
文章:13篇
阅读:4571
文章:15篇
阅读:3268

我要回帖

更多关于 java反序列化 的文章

 

随机推荐