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发现
不是很懂为什么会出现这种情况,希望有大佬能说一下。