序列化NotSerializableException问题

来源:互联网 发布:deepin和ubuntu哪个好 编辑:程序博客网 时间:2024/05/22 05:09

 54powerman
http://54powerman.blog.sohu.com

背景:
某项目中,要将某个自定义类MMessage对象,通过ObjectOutputStream和ObjectInputStream传递,该MMessage的特征描述:
1 该类未继承Serializable接口;
2 其父类Message的父类继承了Serializable接口;
3 其父类中有一个字段类型为java.io.ByteArrayOutputStream类型;
经测试发现,MMessage类序列化过程中,会抛出NotFoundSerializableException,提示如下:
Exception in thread "main" java.io.NotSerializableException: java.io.ByteArrayOutputStream

错误排查:
1 从错误来看,首先想到了看一下MMessage是否继承了Serializable接口,发现其父类的父类继承了Serializable接口,理论上,作为子类的MMessage也应该是可以被序列化的;
2 测试序列化其父类以及父类的父类,发现其父类的父类,可以被序列化,但其父类无法序列化;
3 从上面的分析,问题出在其父类上。再仔细查看错误提示,并分析其父类Message的成员,断定是ByteArrayOutputStream成员无法被序列化(重新定义一个测试用例测试,验证了这一断言)。
4 修改Message类的成员ByteArrayOutputStream buff为transient类型,问题解决。

错误分析:
如果一个类的某个字段不需要被序列化,需要标注该字段为transient类型。适用的情况:
1 该字段为Object,而且该Object不能修改,且不可序列化;
2 该字段为可序列化类型,但它是敏感信息,如密码;
3 该字段为临时变量,如buffer,不需序列化;
例(代码片段):
public class SupperMessage implements Serializable
{
 //......
}
public class Message extends SupperMessage
{
 transient private java.io.ByteArrayOutputStream buff;//此对象无法序列化,且无法更改
 //......
}
public class MMessage extends Message
{
 private String contentId;
 private String user;
 transient private String pass;//敏感信息
 transient private String reserve;//不关心该字段,不需要序列化
 //......
}

类MMessage中标记为transient的成员,不会被序列化。