android對象序列化Parcelable淺析

来源:http://www.cnblogs.com/xerrard/archive/2016/01/20/5144386.html
-Advertisement-
Play Games

一、android序列化簡介我們已經知道在Android使用Intent/Bindler進行IPC傳輸數據時,需要將對象進行序列化。JAVA原本已經提供了Serializable介面來實現序列化,使用起來非常簡單,主要用於對象持久化以及對象的網路傳輸。Serializable開銷比較大,因為序列化和...


一、android序列化簡介

我們已經知道在Android使用Intent/Bindler進行IPC傳輸數據時,需要將對象進行序列化。

JAVA原本已經提供了Serializable介面來實現序列化,使用起來非常簡單,主要用於對象持久化以及對象的網路傳輸。Serializable開銷比較大,因為序列化和反序列化的過程需要大量的I/O操作。

Android提供了Parcelable對象序列化操作是記憶體序列化,主要用於Intent/Bindler的IPC數據傳輸。

 

二、Parcelable序列化使用方法

比如我們使用Parcelable在兩個activity直接通過intent進行傳輸一個Book的對象。

 1 package org.xerrard.demo2;
 2 
 3 import android.os.Parcel;
 4 import android.os.Parcelable;
 5 
 6 /**
 7  * Created by xuqiang on 16-1-20.
 8  */
 9 public class Book implements Parcelable{
10 
11     public String bookname;
12 
13     public Book(String bookname){
14         this.bookname = bookname;
15     }
16 
17     protected Book(Parcel in) {
18         bookname = in.readString();
19     }
20 
21     public static final Creator<Book> CREATOR = new Creator<Book>() {
22         @Override
23         public Book createFromParcel(Parcel in) {
24             return new Book(in);
25         }
26 
27         @Override
28         public Book[] newArray(int size) {
29             return new Book[size];
30         }
31     };
32 
33     @Override
34     public int describeContents() {
35         return 0;
36     }
37 
38     @Override
39     public void writeToParcel(Parcel dest, int flags) {
40         dest.writeInt(bookname);
41     }
42 }

我們需要完成以下幾部。

1. 實現Parcelable介面 
2. 添加實體屬性 
3. 覆寫writeToParcel(Parcel dest, int flags)方法,指定寫入Parcel類的數據。 
4. 創建Parcelable.Creator靜態對象,有兩個方法createFromParcel(Parcel in)與newArray(int size),前者指定如何從Parcel中讀取出數據對象,後者創建一個數組。 
5. 覆寫describeContents方法,預設返回0。 

 

然後我們就可以使用Intent中的putExtra方法將Book對象寫入Intent中,然後使用getExtra方法,就可以從Intent中讀出Book對象。

 

三、Parcelable底層序列化原理

從上面的例子可以看到,Parcelable的序列化方式使用起來還是比較麻煩的。但是,這種方式效率上是比較好的,因為Parcelable的序列化過程是再底層native通過記憶體操作實現的。

詳細的JNI和C/C++底層的記憶體操作可以看這篇文章探索Android中的Parcel機制(上)

摘抄裡面最重要的一句結論

整個讀寫全是在記憶體中進行,主要是通過malloc()、realloc()、memcpy()等記憶體操作進行,所以效率比JAVA序列化中使用外部存儲器會高很多

因此,在IPC過程中,android推薦使用Parcelable序列化方式

 

四:Parcelable的調用關係

我們知道如果要使用Parcelable,必須按照要求實現這五項操作

1. 實現Parcelable介面 
2. 添加實體屬性 
3. 覆寫writeToParcel(Parcel dest, int flags)方法,指定寫入Parcel類的數據。 
4. 創建Parcelable.Creator靜態對象,有兩個方法createFromParcel(Parcel in)與newArray(int size),前者指定如何從Parcel中讀取出數據對象,後者創建一個數組。 
5. 覆寫describeContents方法,預設返回0。 

 這裡面又是怎樣的調用關係呢?

我們看到,writeToParcel是在startActivity的過程中由intent->Bundle->Parcel 一步一步的調用的,然後WriteToParcel會調用native方法,在底層做序列化操作

而createFromParcel是在收到Intent之後,由Intent->Bundle->Parcel 一步一步的調用。

由此可以看出,Parcel的填包解包都是離不開Bundle的。

這裡其實還是有一個疑問,這個Creator是怎麼一回事呢?

我們從源碼中截取Creator這部分來看看。

 1     public final <T extends Parcelable> T readParcelable(ClassLoader loader) {
 2         Parcelable.Creator<T> creator = readParcelableCreator(loader);
 3         if (creator == null) {
 4             return null;
 5         }
 6         if (creator instanceof Parcelable.ClassLoaderCreator<?>) {
 7             return ((Parcelable.ClassLoaderCreator<T>)creator).createFromParcel(this, loader);
 8         }
 9         return creator.createFromParcel(this);
10     }
11 
12     /** @hide */
13     public final <T extends Parcelable> T readCreator(Parcelable.Creator<T> creator,
14             ClassLoader loader) {
15         if (creator instanceof Parcelable.ClassLoaderCreator<?>) {
16             return ((Parcelable.ClassLoaderCreator<T>)creator).createFromParcel(this, loader);
17         }
18         return creator.createFromParcel(this);
19     }
20 
21     /** @hide */
22     public final <T extends Parcelable> Parcelable.Creator<T> readParcelableCreator(
23             ClassLoader loader) {
24         String name = readString(); //此處獲得類名,還不太清楚如何獲得的,如果想深入學習可以再研究
25         if (name == null) {
26             return null;
27         }
28         Parcelable.Creator<T> creator;
29         synchronized (mCreators) {
30             HashMap<String,Parcelable.Creator> map = mCreators.get(loader);
31             if (map == null) {
32                 map = new HashMap<String,Parcelable.Creator>();
33                 mCreators.put(loader, map);
34             }
35             creator = map.get(name);
36             if (creator == null) {
37                 try {
38                     Class c = loader == null ?
39                         Class.forName(name) : Class.forName(name, true, loader);
40                     Field f = c.getField("CREATOR");
41                     creator = (Parcelable.Creator)f.get(null);
42                 }
43                 catch (IllegalAccessException e) {
44                     Log.e(TAG, "Illegal access when unmarshalling: "
45                                         + name, e);
46                     throw new BadParcelableException(
47                             "IllegalAccessException when unmarshalling: " + name);
48                 }
49                 catch (ClassNotFoundException e) {
50                     Log.e(TAG, "Class not found when unmarshalling: "
51                                         + name, e);
52                     throw new BadParcelableException(
53                             "ClassNotFoundException when unmarshalling: " + name);
54                 }
55                 catch (ClassCastException e) {
56                     throw new BadParcelableException("Parcelable protocol requires a "
57                                         + "Parcelable.Creator object called "
58                                         + " CREATOR on class " + name);
59                 }
60                 catch (NoSuchFieldException e) {
61                     throw new BadParcelableException("Parcelable protocol requires a "
62                                         + "Parcelable.Creator object called "
63                                         + " CREATOR on class " + name);
64                 }
65                 catch (NullPointerException e) {
66                     throw new BadParcelableException("Parcelable protocol requires "
67                             + "the CREATOR object to be static on class " + name);
68                 }
69                 if (creator == null) {
70                     throw new BadParcelableException("Parcelable protocol requires a "
71                                         + "Parcelable.Creator object called "
72                                         + " CREATOR on class " + name);
73                 }
74 
75                 map.put(name, creator);
76             }
77         }
78 
79         return creator;
80     }

重點看粗體部分的代碼——真想大白:

在接收端收到parcel之後,解析的時候,會通過反射去獲取對象的Creator,然後保存到一個hashmap中。然後調用Creator的createFromParcel方法來實現解包。

 

反射在源碼中也是無處不在!

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 想通過JQuery來check或者uncheck頁面上的checkbox控制項,我們可能會想到用下麵的代碼:$('#chk-all').on('click', function(){ var checked = $(this).is(':checked'); $("input[type...
  • 在網頁佈局中,經常用到高度自適應增長的佈局。如左側是文章內容部分,隨著文章內容的變化,高度也不固定,右側則是一些固定高度的相關內容。如圖所示: 簡單摸索後發現CSS中有一個min-height的屬性,如果設置min-height為200px,當高度大於200px時,則自動開始增長;小於20...
  • 動畫效果在網站中是一種非常常見的互動式體驗效果,比如側邊欄分享、圖片淡入淡出,我們把這種動畫效果就叫做運動,也就是讓物體動起來。如果想讓一個物體動起來,無非就是改變它的速度,也就是改變屬性值,比如 left、right、width、height、opacity ,那麼既然是運動,就可以分為很多種.....
  • 請選擇本頁面文本看看:http://hovertree.com/h/bjaf/38hq6y9d.htmCSS改變預設文本選中的顏色的方法一般情況下在網頁里的文本我們用滑鼠選中的時候都是藍色的,這個預設顏色也是可以更改的,本文我們學習如何使用CSS3實現改變預設文本選中的顏色。以我的系統舉例(xp 默...
  • 1.何時應當使用margin:需要在border外側添加空白時。空白處不需要背景(色)時。上下相連的兩個盒子之間的空白,需要相互抵消時。如15px + 20px的margin,將得到20px的空白。何時應當時用padding:需要在border內測添加空白時。空白處需要背景(色)時。上下相連的兩個盒...
  • <h2 CSS 定位 (Positioning) 實例</h2 <h3 CSS 實例</h3 CSS 背景實例 CSS 文本實例 CSS 字體(font)實例 CSS 邊框(border)實例 CSS 外邊距 (margin) 實例 CSS 內邊距 (padding) 實例 CSS 列表實例 CS....
  • 隨著CSS3和HTML5的流行,我們的WEB頁面不僅需要更人性化的設計理念,而且需要更酷的頁面特效和用戶體驗。作為開發者,我們需要瞭解一些寶貴的CSS UI開源框架資源,它們可以幫助我們更快更好地實現一些現代化的界面,包括一些移動設備的網頁界面風格設計。本文分享了10個頂級的CSS UI開源框架,有...
  • UIView 不像 UIButton 加了點擊事件就會有點擊效果,體驗要差不少,這裡分別通過自定義和擴展來實現類似 UIButton 的效果。
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...