本文主要介紹Parcelable
和Serializable
的作用、效率、區(qū)別及選擇。
十載的芝罘網(wǎng)站建設(shè)經(jīng)驗(yàn),針對(duì)設(shè)計(jì)、前端、開(kāi)發(fā)、售后、文案、推廣等六對(duì)一服務(wù),響應(yīng)快,48小時(shí)及時(shí)工作處理。網(wǎng)絡(luò)營(yíng)銷推廣的優(yōu)勢(shì)是能夠根據(jù)用戶設(shè)備顯示端的尺寸不同,自動(dòng)調(diào)整芝罘建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無(wú)論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計(jì),從而大程度地提升瀏覽體驗(yàn)。創(chuàng)新互聯(lián)從事“芝罘網(wǎng)站設(shè)計(jì)”,“芝罘網(wǎng)站推廣”以來(lái),每個(gè)客戶項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。
Serializable
的作用是為了保存對(duì)象的屬性到本地文件、數(shù)據(jù)庫(kù)、網(wǎng)絡(luò)流、rmi以方
便數(shù)據(jù)傳輸,當(dāng)然這種傳輸可以是程序內(nèi)的也可以是兩個(gè)程序間的。而Android的Parcelable
的設(shè)計(jì)初衷是因?yàn)?code>Serializable效率過(guò)慢,為了在程序內(nèi)不同組件間以及
不同Android程序間(AIDL)高效的傳輸數(shù)據(jù)而設(shè)計(jì),這些數(shù)據(jù)僅在內(nèi)存中存在,Parcelable
是通過(guò)IBinder
通信的消息的載體。
從上面的設(shè)計(jì)上我們就可以看出優(yōu)劣了
Parcelable
的性能比Serializable
好,在內(nèi)存開(kāi)銷方面較小,所以在內(nèi)存間數(shù)據(jù)傳輸
時(shí)推薦使用Parcelable
,如activity
間傳輸數(shù)據(jù),而Serializable
可將數(shù)據(jù)持久化方便
保存,所以在需要保存或網(wǎng)絡(luò)傳輸數(shù)據(jù)時(shí)選擇Serializable
,因?yàn)閍ndroid不同版本Parcelable
可能不同,所以不推薦使用Parcelable
進(jìn)行數(shù)據(jù)持久化。
對(duì)于Serializable
,類只需要實(shí)現(xiàn)Serializable
接口,并提供一個(gè)序列化版本id(serialVersionUID)
即可。而Parcelable
則需要實(shí)現(xiàn)writeToParcel
、describeContents
函數(shù)以及靜態(tài)的CREATOR變量,實(shí)際上就是將如何打包和解包
的工作自己來(lái)定義,而序列化的這些操作完全由底層實(shí)現(xiàn)。
Parcelable
的一個(gè)實(shí)現(xiàn)例子如下
public class MyParcelable implements Parcelable {
private int mData;
private String mStr;
public int describeContents() {
return 0;
}
// 寫數(shù)據(jù)進(jìn)行保存
public void writeToParcel(Parcel out, int flags) {
out.writeInt(mData);
out.writeString(mStr);
}
// 用來(lái)創(chuàng)建自定義的Parcelable的對(duì)象
public static final Parcelable.Creator CREATOR= new Parcelable.Creator() {
public MyParcelable createFromParcel(Parcel in) {
return new MyParcelable(in);
}
public MyParcelable[] newArray(int size) {
return new MyParcelable[size];
}
};
// 讀數(shù)據(jù)進(jìn)行恢復(fù)
private MyParcelable(Parcel in) {
mData = in.readInt();
mStr = in.readString();
}
}
從上面我們可以看出Parcel的寫入和讀出順序是一致的。如果元素是list讀出時(shí)需要
先new一個(gè)ArrayList
傳入,否則會(huì)報(bào)空指針異常。如下:
list = new ArrayList();
in.readStringList(list);
PS:在自己使用時(shí),read數(shù)據(jù)時(shí)誤將前面int數(shù)據(jù)當(dāng)作long讀出,結(jié)果后面的順序錯(cuò)亂,報(bào)如下異常,當(dāng)類字段較多時(shí)務(wù)必保持寫入和讀取的類型及順序一致。
12-21 20:14:10.317: E/AndroidRuntime(21114): Caused by: java.lan
g.RuntimeException: Parcel android.os.Parcel@4126ed60: Unmarshal
ling unknown type code 3014773 at offset 164
Serializable
序列化不保存靜態(tài)變量,可以使用Transient
關(guān)鍵字對(duì)部分字段不進(jìn)行序
列化,也可以覆蓋writeObject
、readObject
方法以實(shí)現(xiàn)序列化過(guò)程自定義。
更多內(nèi)容詳情請(qǐng)關(guān)注我的GitHub:https://github.com/xiangjiana/Android-MS