十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
本文主要介绍Parcelable
和Serializable
的作用、效率、区别及选择。
Serializable
的作用是为了保存对象的属性到本地文件、数据库、网络流、rmi以方
便数据传输,当然这种传输可以是程序内的也可以是两个程序间的。而Android的Parcelable
的设计初衷是因为Serializable
效率过慢,为了在程序内不同组件间以及
不同Android程序间(AIDL)高效的传输数据而设计,这些数据仅在内存中存在,Parcelable
是通过IBinder
通信的消息的载体。
从上面的设计上我们就可以看出优劣了
Parcelable
的性能比Serializable
好,在内存开销方面较小,所以在内存间数据传输
时推荐使用Parcelable
,如activity
间传输数据,而Serializable
可将数据持久化方便
保存,所以在需要保存或网络传输数据时选择Serializable
,因为android不同版本Parcelable
可能不同,所以不推荐使用Parcelable
进行数据持久化。
对于Serializable
,类只需要实现Serializable
接口,并提供一个序列化版本id(serialVersionUID)
即可。而Parcelable
则需要实现writeToParcel
、describeContents
函数以及静态的CREATOR变量,实际上就是将如何打包和解包
的工作自己来定义,而序列化的这些操作完全由底层实现。
Parcelable
的一个实现例子如下
public class MyParcelable implements Parcelable {
private int mData;
private String mStr;
public int describeContents() {
return 0;
}
// 写数据进行保存
public void writeToParcel(Parcel out, int flags) {
out.writeInt(mData);
out.writeString(mStr);
}
// 用来创建自定义的Parcelable的对象
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];
}
};
// 读数据进行恢复
private MyParcelable(Parcel in) {
mData = in.readInt();
mStr = in.readString();
}
}
从上面我们可以看出Parcel的写入和读出顺序是一致的。如果元素是list读出时需要
先new一个ArrayList
传入,否则会报空指针异常。如下:
list = new ArrayList();
in.readStringList(list);
PS: 在自己使用时,read数据时误将前面int数据当作long读出,结果后面的顺序错乱,报如下异常,当类字段较多时务必保持写入和读取的类型及顺序一致。
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
序列化不保存静态变量,可以使用Transient
关键字对部分字段不进行序
列化,也可以覆盖writeObject
、readObject
方法以实现序列化过程自定义。
更多内容详情请关注我的GitHub:https://github.com/xiangjiana/Android-MS