rialVersionUID是什么?
rialVersionUID有什么作用?
如何生成rialversionuid
java类中rialversionuid 作用 是什么?举个例子说明.
类中影响Serialization进程的特征,两边的操作使用的类版本不同,但它们的 rialVersionUID 必须是一样的。它是用来识别两边的类是否兼容的,两边不同时不应该继续还原状态,而是应该停止下来,因为有人把事情搞错了。如果你的类没有实现 java.io.Serializable 或 java.io.Externalizable,这个字段则没有意义。
如果你没听说过 Java Serialization (序列化,有人书翻译成串行化),那去找些 rialization 介绍看看,下面说的第2段类结构变化时是中级水平的,理解 Java 的一些细节才能理解,多数情况下人们只提到第一种情况(类的结构没有变化时),也只需要第一种情况。
当Serialization两端(比如Socket两端)使用一个类的不同版本时,我们必须提供 rialVersionUID,它可以用JDK自带的 rialver 命令行来计算:
private static final long rialVersionUID = xxxx ;
如果类中出现了下面两个方法,那么将会被用到,否则使用默认的实现:
private void readObject(ObjectInputStream) throws IOException,ClassNotFoundException;
private void writeObject(ObjectOutputStream)throws IOException;
记住这里出现的方法和字段都是 private.
新版本中仅增加了字段或方法而没有改变旧版本中已有的东西时,我们只要保证两个版本中的 rialVersionUID 是一样的就行了.
具体样例可以看 JDK 源码中的像 ArrayList 这些类的代码的 readObject 和 writeObject 方法。
类的结构有些变化时,新版本对旧版本中某些东西进行了删减时, Field 的变化我们需要在readObject和writeObject方法中进行处理
ObjectOutputStream.PutField 类可达到这个目的
只是保证两个版本中的 rialVersionUID 一致是行不通的
类中必需两个常量:
private static final long rialVersionUID;
private static final ObjectStreamField[] rialPersistentFields
下面样例是当客户端和服务端使用的类的版本不同并且类的结构也改变了,比如:
对方使用的类Entry是:
publicclassEntry12...{
privateStringname,email;
privatestaticfinallongrialVersionUID=12L;
}
现在我们改变了类的设计了:
publicclassEntry12...{
privateEntryInternalbasic;
privatestaticfinallongrialVersionUID=12L;
privateclassEntryInternal{
privateStringname,email;
}
}
/*这是一段代码片段,我们假设新版本的类name&email两个字段进行了修改
*(这里我们放到一个内部类EntryInternal中),而旧版本中直接属于这个类.
*
*请注意这里的字段和方法的签名,它们都是private的或者是常量.
*/
publicclassEntry12implementsSerializable{
privateEntryInternalpair=newEntryInternal();
/*这是必要的,因为版本不同*/
privatestaticfinallongrialVersionUID=12L;
**这也是必要的,否则写putFields时会抛出异常.
*Thisfieldwillbeudby{@link#writeObject(ObjectOutputStream)},
*ifthisfieldismissing,follwingexceptionwillbethrewwheninvoke
*{@linkjava.io.ObjectOutputStream.PutField#put(String,)}:
*anexception‘java.lang.IllegalArgumentException:Nosuchobjectfield’willbethrew.
*/
privatestaticfinalObjectStreamField[]rialPersistentFields=newObjectStreamField[]{
newObjectStreamField("name",String.class),//
newObjectStreamField("email",String.class),//
};
/*我们在这里不是直接写出字段,而把要写出的字段包装起来,
我们按需交换字段,而不是直接读写pair这个字段.
*/
privatevoidreadObject(ObjectInputStreaminput)throwsIOException,ClassNotFoundException{
ObjectInputStream.GetFieldgetFields=input.readFields();
/*请注意:使用Serializable进行交换时不使用构造方法,所以这时pair还未初始化.*/
pair=newEntryInternal();
pair.name=(String)getFields.get("name",null);
pair.email=(String)getFields.get("email",null);
}
/*写出时跟读入时一样*/
privatevoidwriteObject(ObjectOutputStreamoutput)throwsIOException{
ObjectOutputStream.PutFieldputFields=output.putFields();
putFields.put("name",pair==null?null:pair.name);
putFields.put("email",pair==null?null:pair.email);
output.writeFields();
}
…..
}
自动生成 rialVersionUID 的设置(2021.1)
自动生成 rialVersionUID 的设置 (本配置2021.1版本配置)
(添加rialVersionUID 的时候,先选中对应的类名,然后按 alt+enter/(ctril+1) 快捷键)没有强调到位。
1:没有设置之前,选中对应的类名,然后按 alt+enter 快捷键 的情况如下所示
2:设置自动生成 rialVersionUID 的方式如下图所示,关键点已逐个标识(java下面)
3:设置之后,选中对应的类名【AppVersion】,然后按 alt+enter(ctril+1) 快捷键的情况如下
4:ok,打完收工
本文发布于:2023-02-28 19:52:00,感谢您对本站的认可!
本文链接:https://www.wtabcd.cn/zhishi/a/167763864372467.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文word下载地址:serialversionuid(serialversionuid的作用).doc
本文 PDF 下载地址:serialversionuid(serialversionuid的作用).pdf
留言与评论(共有 0 条评论) |