一、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.createFr