我使用HashTable<String,Object>
序列化了ObjectOutputStream
个对象。序列化对象时,我没有异常,但在反序列化时,会发生以下异常:
Exception in thread "main" java.io.InvalidClassException: java.lang.Long; local class
incompatible: stream classdesc serialVersionUID = 4290774032661291999, local class
serialVersionUID = 4290774380558885855
当我移除HashTable
中具有非String
值的所有键时,我不再收到错误(我删除的所有键/值对都具有基本类型作为他们的价值)。
可能导致此错误的原因是什么?
更新 - 这是代码
public static String serialize(Quiz quiz) throws IOException{
HashMap<String,Object> quizData = new HashMap<String,Object>();
quizData.put("version", 0); //int
quizData.put("name", quiz.getName()); //String
quizData.put("desc", quiz.getDesc()); //String
quizData.put("timelimitType", quiz.getTimelimitType()); //String
quizData.put("timelimit", quiz.getTimelimit()); //long
ArrayList<String> serializedQuestionsData = new ArrayList<String>();
for (Question question : quiz.getQuestions())
serializedQuestionsData.add(Question.serialize(question));
quizData.put("questions", serializedQuestionsData.toArray(new String[0])); //String[]
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos;
try { oos = new ObjectOutputStream(baos); } catch (IOException error){ throw error; }
try { oos.writeObject(quizData); } catch (IOException error){ throw error; }
return baos.toString();
}
@SuppressWarnings("unchecked")
public static Quiz deserialize(String serializedQuizData) throws IOException, ClassNotFoundException{
ByteArrayInputStream bais = new ByteArrayInputStream(serializedQuizData.getBytes());
ObjectInputStream ois;
try { ois = new ObjectInputStream(bais); } catch (IOException error){ throw error; }
HashMap<String,Object> quizData;
// Exception occurs on the following line!!
try { quizData = (HashMap<String,Object>) ois.readObject(); } catch (ClassNotFoundException error){ throw error; }
Quiz quiz;
if ((int) quizData.get("version") == 0){
quiz = new Quiz((String) quizData.get("name"),
(String) quizData.get("desc"),
(String) quizData.get("timelimitType"),
(long) quizData.get("timelimit"));
for (String serializedQuestionData : (String[]) quizData.get("questions"))
quiz.addQuestion(Question.deserialize(serializedQuestionData));
} else {
throw new UnsupportedOperationException("Unsupported version: \"" + quizData.get("version") + "\"");
}
return quiz;
}
答案 0 :(得分:1)
问题在于您使用toString()
将字节数组输出流转换为String。 toString()
方法仅使用平台默认编码将字节(完全不代表字符,但纯粹是二进制数据)转换为字符串。因此,这是一个有损操作,因为您的平台默认编码没有为每个可能的字节都有一个有效字符。
您不应该使用String来保存二进制数据。字符串包含字符。如果您确实需要String,则使用Hexadecimal或Base64编码器对字节数组进行编码。否则,只需使用字节数组来保存二进制数据:
public static byte[] serialize(Quiz quiz) throws IOException{
...
ByteArrayOutputStream baos = new ByteArrayOutputStream();
...
return baos.toByteArray();
}
@SuppressWarnings("unchecked")
public static Quiz deserialize(byte[] serializedQuizData) throws IOException, ClassNotFoundException{
ByteArrayInputStream bais = new ByteArrayInputStream(serializedQuizData);
...
return quiz;
}
答案 1 :(得分:0)
我能想到的唯一解释是,在您阅读和编写它之间,某些东西正在破坏您的对象流。 “本地类”(4290774380558885855)中的serialVersionID是尝试与Java(tm)兼容的所有Java实现的标准。java.lang.Long
的源代码表明自Java 1.0以来该串行版本ID没有改变。 2。
如果您需要进一步的帮助,您需要提供一个SSCCE,涵盖序列化对象的创建和读取。