序列化:將數(shù)據(jù)結(jié)構(gòu)或者對象轉(zhuǎn)換成二進制串的過程蝴光;
反序列化:將序列化過程中所生成的二進制串轉(zhuǎn)換成數(shù)據(jù)結(jié)構(gòu)或者對象的過程;
持久化:把數(shù)據(jù)結(jié)構(gòu)或?qū)ο筮M行存儲抡医;
序列化的方案主要有:
1躲因、Serializable:java的序列化方案
2、Parcelable:android獨有的
3忌傻、Json/xml/protbuf:廣義的序列化
如何選擇合理的序列化方案:通用性(是否在所有平臺都可以識別大脉,如ios、window系統(tǒng)水孩,linux系統(tǒng))镰矿、健壯性(容錯性,是否容易出錯)俘种、可調(diào)試性可讀性(序列化后對開發(fā)員是否可讀)秤标、性能、可擴展性兼容性宙刘、安全性訪問的限制苍姜;
1.1 Serializable接口: 看源碼實際是一個空接口,是如何實現(xiàn)序列化悬包?
是通過對象的IO流(ObjectOutputStream/writeObject()衙猪、ObjectInputStream/readObject())輔助實現(xiàn)序列化的;
1.2 java中的Externalizable接口:
是需要通過ByteArrayOutputStream包一層布近,然后再像Serializable通過對象IO流(ObjectOutputStream/writeObject()垫释、ObjectInputStream/readObject())序列化和反序列化;
其中必須實現(xiàn)的方法:writeExternal(ObjectOutput out)吊输;readExternal(ObjectInput in)饶号;
實現(xiàn)類必須要求一個無參的構(gòu)造函數(shù);
1.3 android特有的序列化方案Parcelable:
在內(nèi)存中序列化季蚂,使用Binder實現(xiàn)
注意:枚舉類型的序列化茫船,序列化的時候,只會保存枚舉的引用和枚舉的常量扭屁,然后在運行時根據(jù)這些信息查找存在的枚舉類型對象算谈;
Serializable與Parcelable的對比:
Serializable通過IO對硬盤操作,速度較慢料滥,大小不受限制然眼。讀寫的過程大量使用反射,所以會產(chǎn)生內(nèi)存碎片葵腹。
Parcelable直接在內(nèi)存操作高每,效率高屿岂,性能好,一般不能超過1M鲸匿,修改內(nèi)核也只能4M(因為Binder的可用空間一般是1M)爷怀。
有關(guān)思考問題
1、什么是serialVersionUID带欢?如果不定義會發(fā)生什么运授?
serialVersionUID是用于對象的版本控制,是private static final long型的一個ID常量乔煞,通常是對象的哈希碼吁朦;
若不定義,修改了類(如添加新的成員屬性渡贾,或者刪除逗宜、修改)后,serialVersionUID會改變剥啤,從而導(dǎo)致無法再恢復(fù)之前已經(jīng)序列化的對象數(shù)據(jù)锦溪,會丟失之前的數(shù)據(jù),并且會引發(fā)java.io.InvalidClassException無效類異常府怯。
2刻诊、當序列化對象中的 serialVersionUID版本不匹配時,會發(fā)生什么牺丙?
在進行反序列化時则涯,當序列化對象中的 serialVersionUID版本不匹配時會引發(fā)java.io.InvalidClassException無效類異常;
3冲簿、序列化時粟判,如何實現(xiàn)某些成員不進行序列化?
根據(jù)業(yè)務(wù)的需要峦剔,設(shè)置成瞬態(tài)變量(trasient)或者靜態(tài)變量档礁;
4、序列化時吝沫,如果類中包含一個成員未實現(xiàn)可序列化接口呻澜,會發(fā)生什么?
當包含有未實現(xiàn)序列化類的引用時惨险,運行時將引發(fā)不可序列化異常NotSerializableException羹幸。
5、反序列化后的對象辫愉,是否調(diào)用了構(gòu)造函數(shù)重新構(gòu)造栅受?
沒有,因為反序列化后是強轉(zhuǎn)成對象類型的;
序列化前的對象和反序列化后的對象是什么關(guān)系屏镊?(是淺拷貝還是深拷貝依疼?“==”“equal”)
深拷貝,因為前后的地址不同(例外:枚舉)闸衫;
6旬昭、為什么Intent不能直接在組件間傳遞對象而需要通過序列化機制佩番?
因為Activity的通信是需要跟AMS交互的废恋,而AMS是不同的進程忘蟹,屬于跨進程通信的哈雏,所以需要序列化湘纵;
7嘱兼、自定義序列化的過程邦邦?
通過設(shè)置writeObject(ObjectOutputStream out)悦冀、readObject(ObjectInputStream in)方法實現(xiàn)趋翻;
需要自定義序列化的場景:
1、當只有子類實現(xiàn)了序列化接口而父類未實現(xiàn)序列化接口盒蟆,但是在序列化的時候也想將父類的成員屬性進行序列化時踏烙,可以設(shè)置writeObject(ObjectOutputStream out)、readObject(ObjectInputStream in)方法進行設(shè)置實現(xiàn)历等;
2讨惩、當父類實現(xiàn)序列化接口而子類也默認實現(xiàn),此時在序列化過程時寒屯,不想對子類序列化時荐捻,也是設(shè)置writeObject(ObjectOutputStream out)、readObject(ObjectInputStream in)方法中進行拋出NotSerializable異常寡夹;
注意:writeObject处面、readObject方法不是重寫的,而是源碼中是通過反射進行調(diào)用的菩掏。