天天看點

序列化NotSerializableException問題

 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的成員,不會被序列化。

繼續閱讀