Android 序列化 Parcelable VS Seria
Android 的序列化方式 Parcelable
Parcel 介绍:Parcel 内部包装了可序列化的数据,可以在 Binder 中自由传输
Parcelable 是一个接口,只要实现这个接口,一个类的对象就可以实现序列化并可以通过 Intent 和 Binder 传递。
package com.renxl.touchevent.scrollview;
import android.os.Parcel;
import android.os.Parcelable;
public class Book implements Parcelable {
private String name;
private int price;
private Category category;
protected Book(Parcel in) {
name = in.readString();
price = in.readInt();
category = in.readParcelable(Category.class.getClassLoader());
}
public static final Creator<Book> CREATOR = new Creator<Book>() {
@Override
public Book createFromParcel(Parcel in) {
return new Book(in);
}
@Override
public Book[] newArray(int size) {
return new Book[size];
}
};
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(name);
dest.writeInt(price);
dest.writeParcelable(category, flags);
}
}
由上段代码可以看出序列化过程需要实现的功能有 序列化,反序列化,内容描述 三部分
一、序列化功能
序列化功能是由 writeToParcel 方法完成的,是通过 Parcel 的一些了 write 方法完成的。
注意:序列化的类的成员属性也必须是可序列化的。Parcel 的 writeParcelable 方法中除了可序列化的属性,还需要添加 int 类型的 flag 参数,表示该对象应以何种方式写入,一般传 writeToParcel 方法的 flag 参数或者0。
二、反序列化功能
反序列化过程由 CREATOR 来完成,其内部表明了如何创建序列化对象和序列化对象的数组,创建序列化对象使用了序列化对象的参数为 Parcel 的构造方法,该方法中根据序列化过程得到的 Parcel 对象的 read 系列方法将序列化对象的内容还原完成反序列化过程。
如果序列化类的属性有可序列化对象,在反序列化过程中需要传递当前线程的上下文类加载器,否则会报无法找到类的错误。
三、内容描述功能
内容描述功能由 describeContents 完成,几乎所有情况该方法都返回 0 ,仅当当前对象中存在的文件描述符时此方法返回 1 。
文件描述符: 文件描述符在形式上是一个非负整数。实际上,它是一个索引值,指向内核为每一个进程所维护的该进程打开文件的记录表。当程序打开一个现有文件或者创建一个新文件时,内核向进程返回一个文件描述符。在程序设计中,一些涉及底层的程序编写往往会围绕着文件描述符展开。但是文件描述符这一概念往往只适用于UNIX、Linux这样的操作系统。
Android 系统提供的实现了 Parcelable 的类,Intent,Bundle,Bitmap等,List 和 Map 也可以序列化,前提是里面的内容都可以序列化。
序列化方式 Serializable 接口
Java 中提供的序列化方式,实现 Serializable 接口,是一个空接口,为对象提供了标准的序列化和反序列化操作。
只需要在类的声明中指定 private static final long serialVersionUID = 123423432234L 即可,该值并不是必须的
serialVersionUID 作用为辅助序列化和反序列化过程,反序列化时序列化数据中的 serialVersionUID 和当前类的 serialVersionUID 相同是才能成功,不相同则会反序列化是吧,程序 crash。类中该值不指定时可以根据当前类的结构自动生成 hash ,不过当类结构变化时反序列化会失败。如果指定,就算类的结构发生改变也会最大限度的恢复数据。
还有两点点需要注意
- 静态成员变量属于类不属于对象,所以不参与序列化过程,反序列化得到的对象也可以使用该属性,但原理是因为使用的是类的静态变量
- 被 transient 修饰的成员不参与序列化,反序列化后也不会得到 transient 修饰的成员
Parcelable 和 Serializable 区别
- 两者都可以实现序列化并且都可以用于 Intent 间数据传递
- Serializable 是 Java 中的序列化接口,使用简单但是开销很大,序列化和反序列化过程需要很多 I/O 操作
- Parcelable 是 Android 中的序列化方式,更适合用在 Android 中,确定是使用麻烦,但是效率很高。Android 中首选 Parcelable
- Parcelable 主要用于 Android 跨进程通信时在对内存数据的序列话,跨进程传输的数据是必须序列化的,Parcelable 更方便
- 将对象序列化到设备的本地文件或者网络传输,建议使用 Serializable,