您的位置:首页 > 其它

序列化和反序列化时的注意点

2015-08-27 08:41 323 查看
如果一个对象不止一次写入对象流,并不会存储对象的多个副本。第一次写入一个对象时,就会为它创建一个序列号。Java虚拟机将对象的所有内容和序列号一起写入对象流。如果再写入相同对象(注意:这里的相同指的是相同的引用(地址),具有相同的引用则表示是相同的对象或者可以说是同一个对象),就只存储序列号。如果中途更改了对象的属性,那么该对象的内容是不会存储在流中的,只存储序列号。所以当需要从流中读出第二次写入的对象是行不通的。为了避免这个问题,可以在写的时候写入一个克隆的对象(使用clone()方法),这样它们的引用不相同,就会都写进流中了。

    序列化运行时使用一个称为 serialVersionUID 的版本号与每个可序列化类相关联,该序列号在反序列化过程中用于验证序列化对象的发送者和接收者是否为该对象加载了与序列化兼容的类。如果接收者加载的该对象的类的 serialVersionUID 与对应的发送者的类的版本号不同,则反序列化将会导致 
InvalidClassException
。可序列化类可以通过声明名为 
"serialVersionUID"
 的字段(该字段必须是静态
(static)、最终 (final) 的 
long
 型字段)显式声明其自己的 serialVersionUID:
ANY-ACCESS-MODIFIER static final long serialVersionUID = 42L;

如果可序列化类未显式声明 serialVersionUID,则序列化运行时将基于该类的各个方面计算该类的默认 serialVersionUID 值,如“Java(TM) 对象序列化规范”中所述。不过,强烈建议 所有可序列化类都显式声明 serialVersionUID 值,原因是计算默认的 serialVersionUID 对类的详细信息具有较高的敏感性,根据编译器实现的不同可能千差万别,这样在反序列化过程中可能会导致意外的
InvalidClassException
。因此,为保证
serialVersionUID 值跨不同 java 编译器实现的一致性,序列化类必须声明一个明确的 serialVersionUID 值。还强烈建议使用 
private
 修饰符显示声明 serialVersionUID(如果可能),原因是这种声明仅应用于直接声明类 -- serialVersionUID
字段作为继承成员没有用处。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: