Serializable和Parcelable是什么?

Serializable和Parcelable都是一种将对象序列化的接口(序列化,表示将一个对象转换成可存储或可传输的状态。序列化后的对象可以在网络上进行传输,也可以存储到本地),Serializable是java中提供的,而Parcelable是Android中提供的。

准备工作

写一个User类,下面会分别使这个类接入Serializable和Parcelable接口。

public class User {
    private static final String TAG = "User";
    private String name;
    private int age;
//为了省略篇幅,构造方法,getter和setter省略不写了
}

在MainActivity的onCreate()方法中加入下面的代码

Intent intent = new Intent(this, SecondActivity.class);
intent.putExtra("user", new User("test", 123));
startActivity(intent);

创建一个SecondActivity,同样在onCreate()方法中加入下面的代码

User user = (User) getIntent().getSerializableExtra("user");
Log.i(TAG, user.getName() + "   " + user.getAge() + "");

Serializable的使用

在完成了上面的代码后,你会发现MainActivity里如图所示的这一行报错了,这是因为intent不能直接put对象进去,需要对象实现Serializable或者Parcelable接口才可以。


这里我们直接在User类的声明后加上 implements Serializable就可以了,之后就发现不再报错,运行程序,查看log,发现User对象成功传递了。

Parcelable的使用

这次我们让User类实现Parcelable接口,接入接口之后我们需要重写两个方法。

public class User implements Parcelable {
    private static final String TAG = "User";
    private String name;
    private int age;
//同样省略了构造方法,getter和setter
    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(getName());
        dest.writeInt(getAge());
    }
}

describeContents()返回当前对象的内容描述。如果含有文件描述符,返回1,否则返回0,几乎所有的情况都返回0。
writeToParcel()是将当前对象写入一个Parcel中,一般是将当前对象的属性值通过writexxxx()方法写入。上面的代码就是分别将User对象的name和age写入。
除了上面的,我们还需要写一个Creator用来返回User对象,写法如下边所示,一般newArray()的返回值都是new User[size]。
在createFromParcel()方法里你可以直接写return new User(source.readString(),source.readInt());,但更标准的写法是再写一个构造方法,这个构造方法的参数为Parcel对象,然后在这个构造方法里对对象的属性进行赋值。
需要注意的是,当一个对象中有多个同种类型的变量时,通过readxxxx()方法获取的值和你writexxxx()方法写入的顺序是一致,比方说我现在有一个类只有三个int类型的变量,分别为num1,num2,num3,然后我在 writeToParcel()方法里依次写入了num1、2、3,那么我的三个read的值也分别是num1、2、3。

    public static final Creator<User> CREATOR = new Creator<User>() {
        @Override
        public User createFromParcel(Parcel source) {
            return new User(source);
        }

        @Override
        public User[] newArray(int size) {
            return new User[size];
        }
    };

    private User(Parcel in) {
        this.name = in.readString();
        this.age = in.readInt();
    }

我们把Second里的User对象的声明改为User user = getIntent().getParcelableExtra("user");,运行程序,发现程序成功运行。

Serializable和Parcelable的区别

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

一个有意思的现象

既然Serializable与Parcelable都是接口,那么说明我们的一个类可以同时接入这两个接口,当我们这么做时,会发生什么现象呢?
我们使User类同时接入这两个接口,并在writeToParcel()方法里加一句logLog.i(TAG, "used writeToParcel");
这时我们发现MainActivity的putExtra()方法报错,因为这时编译器无法判断到底是使用Serializable还是Parcelable,在这里我们为了验证这个有意思的现象,把User强转为Serializable,并且在SecondActivity里也用Serializable接收。


运行程序,发现一个有意思的现象出现了,我们同样能够接收到对象,但是通过筛选log发现

不是很懂为什么会出现这种情况,希望有大佬能说一下。
收藏
评论加载中...