[英]Finding serialVersionUID of serialized object
有没有办法确定生成的序列化Java对象的serialVersionUID
?
问题是我在没有明确指定serialVersionUID
情况下序列化了一个对象。 现在反序列化过程抱怨类不兼容。 但是我并没有以一种使它不兼容的方式改变这个类。 所以我假设在类中指定serialVersionUID
就足够了,因为它存储在对象数据中。 为此,我需要从序列化数据中读取serialVersionUID
。
有一种简单的方法可以找到一个类的serialversionUID-
假设你有一个你忘记提及serialversionUID的课程 -
import java.io.Serializable;
public class TestSerializable implements Serializable {
}
这样做 -
serialver -classpath . TestSerializable
这印刷品 -
static final long serialVersionUID = 5832063776451490808L;
serialver是一个与JDK一起提供的实用程序
这对我有用:
long serialVersionUID = ObjectStreamClass.lookup(YourObject.getClass()).getSerialVersionUID();
希望它也能帮到你。
您可以通过扩展ObjectInputStream来完成此操作:
public class PrintUIDs extends ObjectInputStream {
public PrintUIDs(InputStream in) throws IOException {
super(in);
}
@Override
protected ObjectStreamClass readClassDescriptor() throws IOException,
ClassNotFoundException {
ObjectStreamClass descriptor = super.readClassDescriptor();
System.out.println("name=" + descriptor.getName());
System.out.println("serialVersionUID=" + descriptor.getSerialVersionUID());
return descriptor;
}
public static void main(String[] args) throws IOException,
ClassNotFoundException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
List<Object> list = Arrays.asList((Object) new Date(), UUID.randomUUID());
oos.writeObject(list);
oos.close();
InputStream in = new ByteArrayInputStream(baos.toByteArray());
ObjectInputStream ois = new PrintUIDs(in);
ois.readObject();
}
}
我相信通过替换方法返回的描述符可以读取所有序列化数据,但我还没有尝试过。
存在与序列化位相关联的元数据(如果您愿意,可以使用标题)。 如果您知道它在哪个位置( SerialVersionUID
与其他信息(如类名)一起写入),您可以从元数据中读取值。
我认为这篇文章可能对您有所帮助: Java序列化算法揭示了 。
请注意,这些位是“明确地”写入的(除非您明确加密了流),因此您可能需要使用HEX编辑器来查看SerialVersionUID
是什么。
对象的序列化有一个指定的语法:
请参阅http://java.sun.com/javase/6/docs/platform/serialization/spec/serialTOC.html中的第6.4章
使用它,您应该能够确定序列化对象的SerialVersionUID。
这正是你应该做的 - 指定你自己的static final long serialVersionUID
。
在Serializable的文档中有一节介绍它。
除非你已经指定了serialVersionUID
否则我不相信除了解密流之外还有一个简单的方法来解决它,因为@WMR建议。
有更简单的方法来获取序列化对象的serialVersionUID - 只需使用apache commons serialization utils将其反序列化为具有不同串行版本uid的相同类,并读取或多或少的异常消息:
org.apache.commons.lang.SerializationException:java.io.InvalidClassException:my.example.SerializableClass; 本地类不兼容:stream classdesc serialVersionUID = -1,本地类serialVersionUID = -2
获取它的最简单方法(特别是在难以找到的情况下)是为serialVersionUID定义任何数字并在反序列化失败时监视堆栈跟踪,它将打印出原始数据。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.