java.io.invalidclassexception: failed to read class descriptor

此文来自 :   马开东博客  转载请注明出处 网址  http://www.makaidong.com
摘要:  一个netty序列化引起的思考。
"一个netty序列化引起的思考"

今天在使用netty 传输网络对象报了异常错误 java.io.invalidclassexception: failed to read class descriptor

自己捣鼓了许久。

后来查了下api invalidclassexception

api文档描述

当 serialization 运行时检测到某个类具有以下问题之一时,抛出此异常。

  • 该类的序列版本号与从流中读取的类描述符的版本号不匹配
  • 该类包含未知数据类型
  • 该类没有可访问的无参数构造方法

然后看了下netty client和netty server bean 信息

client 代码

public class document implements serializable   {
    private static final long serialversionuid = 8213761673743652497l;

。。。。。。。。。。。。
}

server bean 代码

public class document implements serializable   {
    private static final long serialversionuid = 1876613896675121682l;
。。。。。。。。。。。。
}

仔细看下 不难发现 2个 bean的 serialversionuid 不一致

然后google一下 序列化id(serialversionuid)

扫描二维码关注公众号,回复: 575297 查看本文章

下面对序列化id 的介绍

简单来说,java的序列化机制是通过在运行时判断类的serialversionuid来验证版本一致性的。在进行反序列化时,jvm会把传来的字节流中的serialversionuid与本地相应实体(类)的serialversionuid进行比较,如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常。

当实现java.io.serializable接口的实体(类)没有显式地定义一个名为serialversionuid,类型为long的变量时,java序列化机制会根据编译的class自动生成一个serialversionuid作序列化版本比较用,这种情况下,只有同一次编译生成的class才会生成相同的serialversionuid 。

 

如果我们不希望通过编译来强制划分软件版本,即实现序列化接口的实体能够兼容先前版本,未作更改的类,就需要显式地定义一个名为serialversionuid,类型为long的变量,不修改这个变量值的序列化实体都可以相互进行串行化和反串行化。

 

顿时幡然大悟,应该是serialversionuid 不一致而且又不是同一次编译生成的class 。从而导致server端反序列化失败。

于是想到了解决办法。让netty client 和 netty server 共享一个document 对象即解决了问题。

+

+

+

=

=

=

猜你喜欢

转载自fantaxy025025.iteye.com/blog/2326612