Serializable和Parcelable

区别

1、作用

Serializable的作用是为了保存对象的属性到本地文件、数据库、网络流、rmi以方便数据传输,当然这种传输可以是程序内的也可以是两个程序间的。Serializable原理: 反射、递归。而Android的Parcelable的设计初衷是因为Serializable效率过慢,为了在程序内不同组件间以及不同Android程序间(AIDL)高效的传输数据而设计,这些数据仅在内存中存在,Parcelable是通过IBinder通信的消息的载体。

从上面的设计上我们就可以看出优劣了。

2、效率及选择

Parcelable的性能比Serializable好,在内存开销方面较小,所以在内存间数据传输时推荐使用Parcelable,如activity间传输数据,而Serializable可将数据持久化方便保存,所以在需要保存或网络传输数据时选择Serializable
,因为android不同版本Parcelable可能不同,所以不推荐使用Parcelable进行数据持久化

Serializable的实现

对于Serializable,类只需要实现Serializable接口,并提供一个序列化版本id(serialVersionUID)即可。

public class MySerializable implements Serializable{
    private static final long serialVersionUID = 1L;
    private int mData;
    private String mStr;

    public int getmData() {
        return mData;
    }

    public void setmData(int mData) {
        this.mData = mData;
    }

    public String getmStr() {
        return mStr;
    }

    public void setmStr(String mStr) {
        this.mStr = mStr;
    }
}

关于Serializable的serialVersionUID - SMCwwh - 博客频道 - CSDN.NET

Parcelable的实现

  • 步骤 1:自定义实体类,实现Parcelable接口,重写其两个方法。
  • 步骤2:该实体类必须添加一个常量CREATOR(名字大小写都不能使其他的),该常量必须实现Parcelable的内部接口:Parcelable.Creator,并实现该接口中的两个方法。
public class MyParcelable implements Parcelable {
    private int mData;
    private String mStr;

    // 将对象中的属性保存至目标对象dest中  
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(mData);
        dest.writeString(mStr);
    }

    //用来创建自定义的Parcelable的对象
    public static final Creator<MyParcelable> CREATOR = new Creator<MyParcelable>() {
        @Override
        public MyParcelable createFromParcel(Parcel in) {
            return new MyParcelable(in);
        }

        //重写createFromParcel方法,创建并返回一个获得了数据的MyParcelable对象  
        public MyParcelable[] newArray(int size) {
            return new MyParcelable[size];
        }
    };

    // 读数据进行恢复,带参构造器方法私用化,本构造器仅供类的方法createFromParcel调用  
    protected MyParcelable(Parcel in) {
        mData = in.readInt();
        mStr = in.readString();
    }

    @Override
    public int describeContents() {
        return 0;
    }

    public String getmStr() {
        return mStr;
    }

    public void setmStr(String mStr) {
        this.mStr = mStr;
    }

    public int getmData() {
        return mData;
    }

    public void setmData(int mData) {
        this.mData = mData;
    }
}

从上面我们可以看出Parcel的写入和读出顺序是一致的。如果元素是list读出时需要先new一个ArrayList传入,否则会报空指针异常。如下:

list = new ArrayList<String>();
in.readStringList(list);

PS: 在自己使用时,read数据时误将前面int数据当作long读出,结果后面的顺序错乱,报如下异常,当类字段较多时务必保持写入和读取的类型及顺序一致。

11-21 20:14:10.317: E/AndroidRuntime(21114): Caused by: java.lang.RuntimeException: Parcel android.os.Parcel@4126ed60: Unmarshalling unknown type code 3014773 at offset 164

其他

Serializable序列化不保存静态变量,可以使用Transient关键字对部分字段不进行序列化,也可以覆盖writeObject、readObject方法以实现序列化过程自定义

引用:
Android Parcelable和Serializable的区别 - djun100的专栏 - 博客频道 - CSDN.NET
关于Serializable的serialVersionUID - SMCwwh - 博客频道 - CSDN.NET
实现Serializable的单例模式 | 学步园
测试—Android系统中Parcelable和Serializable的区别 - 泡在网上的日子
Java Serializable系列化与反系列化 - SMCwwh - 博客频道 - CSDN.NET

Android Serializable与Parcelable原理与区别 - 每天记录点滴 - 博客频道 - CSDN.NET
http://blog.csdn.net/androiddevelop/article/details/22108843?utm_source=tuicool&utm_medium=referral
ArrayList源码分析——如何实现Serializable - - ITeye技术网站
http://java-road-126-com.iteye.com/blog/1463313
探索Android中的Parcel机制(上) - 文斌的专栏 - 博客频道 - CSDN.NET
http://blog.csdn.net/caowenbin/article/details/6532217
探索Android中的Parcel机制(下) - 文斌的专栏 - 博客频道 - CSDN.NET
http://blog.csdn.net/caowenbin/article/details/6532238
Android中的Parcel是什么 - nkmnkm的专栏 - 博客频道 - CSDN.NET
http://blog.csdn.net/niu_gao/article/details/6451699

高级—Java 序列化的高级认识 - 开源中国社区

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值