Android 序列化 Parcelable VS Serializable

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 CREATOR = new Creator() {
            @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 ,不过当类结构变化时反序列化会失败。如果指定,就算类的结构发生改变也会最大限度的恢复数据。

还有两点点需要注意

  1. 静态成员变量属于类不属于对象,所以不参与序列化过程,反序列化得到的对象也可以使用该属性,但原理是因为使用的是类的静态变量
  2. 被 transient 修饰的成员不参与序列化,反序列化后也不会得到 transient 修饰的成员

Parcelable 和 Serializable 区别

  1. 两者都可以实现序列化并且都可以用于 Intent 间数据传递
  2. Serializable 是 Java 中的序列化接口,使用简单但是开销很大,序列化和反序列化过程需要很多 I/O 操作
  3. Parcelable 是 Android 中的序列化方式,更适合用在 Android 中,确定是使用麻烦,但是效率很高。Android 中首选 Parcelable
  4. Parcelable 主要用于 Android 跨进程通信时在对内存数据的序列话,跨进程传输的数据是必须序列化的,Parcelable 更方便
  5. 将对象序列化到设备的本地文件或者网络传输,建议使用 Serializable,

你可能感兴趣的:(Android 序列化 Parcelable VS Serializable)