在 Android 开发中,我们经常会使用序列化的方式存储一些数据信息,常用的序列化方式包括 Serializable 和 Parcelable 两种。相比较而言,Serializable 实现更简单,而 Parcelable 性能更高。
使用 Parcelable 时,需要在数据类实现 Parcelable 接口,一般情况下,要实现该接口声明的如下方法:
fun writeToParcel(parcel: Parcel, flags: Int)
fun describeContents(): Int
另外,为了反序列化构建数据类对象,还需要实现单例 CREATOR 变量。
companion object CREATOR : Parcelable.Creator<Test> {
override fun createFromParcel(parcel: Parcel): Test {
return Test(parcel)
}
override fun newArray(size: Int): Array<Test?> {
return arrayOfNulls(size)
}
}
为了简化开发复杂度,精简代码,Kotlin 官方提供了 Parcelize 注解,只需要在实现了 Parcelable 接口的数据类上添加此注解,注解解析器就会在编译时动态生成 Parcelable 接口中声明的方法,无需开发人员自己实现。
import kotlinx.android.parcel.Parcelize
@Parcelize
data class Test : Parcelable {
// 不需要实现
// override fun writeToParcel(parcel: Parcel, flags: Int) {}
// override fun describeContents(): Int {}
// companion object CREATOR
}
但Parcelize 注解只是提供了最简单的处理方式,在特定场景下,使用该注解可能会遇到无法正常序列化存取数据的问题。
@Parcelize
data class Test(
val id: Int
) : Parcelable {
var name: String
constructor(pp: TestNew) : this(
pp.id
) {
name = pp.name
}
}
在如上数据类 Test 中,包含两个成员变量,其中 id 通过构造方法定义,而 name 没有。两个构造方法,一个默认构造方法,接收参数 id;一个写在类内的扩展构造方法,接收参数 pp。
则 Parcelize 注解动态生成的方法在序列化该类对象时,只会处理默认构造方法中的变量,即 id,而不会处理 name。而且只会通过默认构造方法序列化存取对象,并不会通过扩展构造方法序列化存取对象。即会造成 name 数据缺失。
如果需要序列化存取所有变量,只能开发人员自己实现 Parcelable 接口的所有方法。
data class Test(
val id: Int
) : Parcelable {
var name: String
constructor(pp: TestNew) : this(
pp.id
) {
name = pp.name
}
constructor(parcel: Parcel) : this(
parcel.readInt()
) {
name = parcel.readString()
}
override fun writeToParcel(parcel: Parcel, flags: Int) {
parcel.writeInt(id)
parcel.writeString(name)
}
override fun describeContents(): Int {
return 0
}
companion object CREATOR : Parcelable.Creator<Test> {
override fun createFromParcel(parcel: Parcel): Test{
return Test(parcel)
}
override fun newArray(size: Int): Array<Test?> {
return arrayOfNulls(size)
}
}
}
|