在做Android開發的過程中,序列化是非常常見的。比如要將對象保存本地磁盤或者在網絡上傳輸等。實現序列化有兩種方式,一種是實現Serializable接口,第二種是實現Parcelable。
Serializable與Parcelable的區別
1、Serializable是JDK提供的接口,而Parcelable是Android SDK提供的。
2、Serializable序列化是基於磁盤的,而Parcelable是基於內存的。在內存中讀寫肯定效率要高於磁盤,所以Android中跨進程傳遞對象都是使用Parcelable。
Parcelable接口定義
1 public interface Parcelable { 2 //內容描述接口,基本不用管 3 public int describeContents(); 4 //寫入接口函數,打包 5 public void writeToParcel(Parcel dest, int flags); 6 //讀取接口,目的是要從Parcel中構造一個實現了Parcelable的類的實例處理。因為實現類在這裡還是不可知的,所以需要用到模板的方式,繼承類名通過模板參數傳入。 7 //為了能夠實現模板參數的傳入,這裡定義Creator嵌入接口,內含兩個接口函數分別返回單個和多個繼承類實例。 8 public interface Creator<T> { 9 public T createFromParcel(Parcel source); 10 public T[] newArray(int size); 11 }
從parcelable接口定義中,我們可以看到,實現parcelable接口,需要我們實現下面幾個方法:
1.describeContents方法。內容接口描述,默認返回0就可以;
2.writeToParcel 方法。該方法將類的數據寫入外部提供的Parcel中.即打包需要傳遞的數據到Parcel容器保存,以便從parcel容器獲取數據,該方法聲明如下:
writeToParcel (Parcel dest, int flags) 具體參數含義見javadoc
3.靜態的Parcelable.Creator接口,本接口有兩個方法:
createFromParcel(Parcel in) 從Parcel容器中讀取傳遞數據值,封裝成Parcelable對象返回邏輯層。
newArray(int size) 創建一個類型為T,長度為size的數組,僅一句話(return new T[size])即可。方法是供外部類反序列化本類數組使用。
Parcelable的使用
1 public class AppContent implements Serializable, Parcelable { 2 //應用名字 3 private String name; 4 //應用下載鏈接 5 private String url; 6 7 private int downloadPercent = 0; 8 9 private Status status = Status.PENDING; 10 11 public AppContent(String name, String url) { 12 this.name = name; 13 this.url = url; 14 } 15 16 public String getName() { 17 return name; 18 } 19 20 public void setName(String name) { 21 this.name = name; 22 } 23 24 public String getUrl() { 25 return url; 26 } 27 28 public void setUrl(String url) { 29 this.url = url; 30 } 31 32 public int getDownloadPercent() { 33 return downloadPercent; 34 } 35 36 public void setDownloadPercent(int downloadPercent) { 37 this.downloadPercent = downloadPercent; 38 } 39 40 public Status getStatus() { 41 return status; 42 } 43 44 public void setStatus(Status status) { 45 this.status = status; 46 } 47 48 @Override 49 public String toString() { 50 return name; 51 } 52 53 @Override 54 public int describeContents() { 55 return 0; 56 } 57 58 //實現Parcel接口必須覆蓋實現的方法 59 @Override 60 public void writeToParcel(Parcel dest, int flags) { 61 /*將AppContent的成員寫入Parcel, 62 * 注:Parcel中的數據是按順序寫入和讀取的,即先被寫入的就會先被讀取出來 63 */ 64 dest.writeString(name); 65 dest.writeString(url); 66 dest.writeInt(downloadPercent); 67 dest.writeValue(status); 68 } 69 70 //該靜態域是必須要有的,而且名字必須是CREATOR,否則會出錯 71 public static final Parcelable.Creator<AppContent> CREATOR = 72 new Parcelable.Creator<AppContent>() { 73 74 @Override 75 public AppContent createFromParcel(Parcel source) { 76 //從Parcel讀取通過writeToParcel方法寫入的AppContent的相關成員信息 77 String name = source.readString(); 78 String url = source.readString(); 79 int downloadPercent = source.readInt(); 80 Status status = (Status)source.readValue(new ClassLoader(){}); 81 AppContent appContent = new AppContent(name, url); 82 appContent.setDownloadPercent(downloadPercent); 83 appContent.setStatus(status); 84 //更加讀取到的信息,創建返回Person對象 85 return appContent; 86 } 87 88 @Override 89 public AppContent[] newArray(int size) 90 { 91 // TODO Auto-generated method stub 92 //返回AppContent對象數組 93 return new AppContent[size]; 94 } 95 }; 96 97 }
通過Intent進行傳遞:
1 Intent intent = new Intent(Constants.DOWNLOAD_MSG); 2 Bundle bundle = new Bundle(); 3 bundle.putParcelable("appContent", appContent); 4 intent.putExtras(bundle);
更多Android相關信息見Android 專題頁面 http://www.linuxidc.com/topicnews.aspx?tid=11