54powerman
<a href="http://54powerman.blog.sohu.com">http://54powerman.blog.sohu.com</a>
背景:
某項目中,要将某個自定義類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的成員,不會被序列化。