使用带有 Kotlin 数据 class 的 Parceler 和用于序列化的构造函数

Use of Parceler with Kotlin data class with constructor for serialization

有没有办法在不对每个字段使用 @ParcelProperty 注释的情况下,将 Parceler 与 Kotlin 数据 classes 和构造函数一起用于序列化?

如果我尝试像这样使用库:

@Parcel
data class Valve @ParcelConstructor constructor(val size: Int)

我得到 Error:Parceler: No corresponding property found for constructor parameter arg0。但是如果我添加 @ParcelProperty("size") 它就可以正常工作。
这是为什么?

更新:
还有其他方法可以使用这个库。
我可以只删除 @ParcelConstructor 注释,但我会得到错误
Error:Parceler: No @ParcelConstructor annotated constructor and no default empty bean constructor found.
我想(还没有测试过)我也可以使所有构造函数参数可选并添加 @JvmOverloads 但这有一个副作用,我必须检查 class 的所有属性是否为空.

更新 2:
这对我有用:

@Parcel
data class Valve(val size: Int? = null)

简而言之,生成的 Java class 必须具有默认的空构造函数。实现这一目标的一种方法是按照上述方式进行 - 所有变量都应具有默认值。

根据文档,Parceler 默认使用 public 字段。但是通常的 Kotlin data class(如您的示例所示)更像是 "traditional getter/setter bean",因为每个 Kotlin 属性 都由一个私有字段和一个 getter/[setter].

TL; DR:我认为这会起作用:

@Parcel(Serialization.BEAN)
data class Valve(val size: Int = 10)

注意默认值,它允许 Kotlin 自动生成一个额外的空构造函数,这是 Java 规范所要求的。

另一种方法是标记我们已有的构造函数:

@Parcel(Serialization.BEAN)
data class Driver @ParcelConstructor constructor(val name: String)

具体文档:https://github.com/johncarl81/parceler#gettersetter-serialization

我知道这个问题已经有了答案,但对于那些也在努力让 Parceler 与 kotlin 数据对象一起工作的未来观众,我编写了一个新的注释处理器来为 Kotlin 数据生成 Parcelable 样板 class是的。它旨在大量减少使您的数据 classes Parcelable:

的样板代码

https://github.com/grandstaish/paperparcel

用法:

@PaperParcel 注释您的数据 class,实施 PaperParcelable,并添加生成的 CREATOR 的 JVM 静态实例,例如:

@PaperParcel
data class Example(
  val test: Int,
  ...
) : PaperParcelable {
  companion object {
    @JvmField val CREATOR = PaperParcelExample.CREATOR
  }
}

现在你的数据class是Parcelable,可以直接传递给BundleIntent

编辑:更新最新API

只需添加默认构造函数:

@Parcel
data class Valve(val size: Int) {
    constructor() : this(0)
}

如果您使用 Kotlin 1.1.4 或更高版本,使用 @Parcelize 注释会更容易

为此,首先将其添加到 build.gradle

android {
    //other codes

    //for using latest experimental build of Android Extensions
    androidExtensions {
        experimental = true
    }
}

然后像这样改变你的 class

@Parcelize
data class Valve(val size: Int? = null) : Parcelable