使用 Moshi 反序列化可以是两种数据类型之一的字段

tat*_*000 6 java android json kotlin moshi

我从 OrientDB 服务器收到一些 JSON,看起来像这样:

{
    ...
    "out": ...,
    "in": ...,
    ...
}
Run Code Online (Sandbox Code Playgroud)

现在,这两个领域out,并in可以是两种类型之一:String和我自己的自定义对象(我们称之为Record)。例如,对于一个请求,我可能会收到以下信息:

{
    ...
    "out": "#17:0",
    "in": {
        ...
    },
    ...
}
Run Code Online (Sandbox Code Playgroud)

对于另一个我可能会得到:

{
    ...
    "out": {
        ...
    },
    "in": "#18:2",
    ...
}
Run Code Online (Sandbox Code Playgroud)

等等。两者都可能是Strings,两者都可能是Records,一个可能是 a String,另一个是 a Record,等等等等。现在,当我使用Moshi反序列化这种 JSON 时,我将有两个参数outin保存它们各自键的值;但是,因为这些值不是固定的数据类型,所以说起来容易做起来难。

创建多个 POJO(或“POKO”,我猜,因为我使用的是 Kotlin)是行不通的,因为这些对象可以在其他JSON 对象和类似的东西中找到。我需要一个对象,这些参数可以采用可变数据类型。那我该怎么做呢?

我是否必须在 Moshi 中编写一个自定义适配器来序列化/反序列化这些值?如果是这样,我将如何编写可以根据参数值分配某种数据类型的程序?或者我可以找到/编写某种可以容纳两种可能数据类型的 Kotlin 类/函数/扩展函数吗?

如果相关,我还使用 Retrofit 2 + RxJava 2 来异步进行 HTTP 调用,所以如果这些库中有任何数据类型或函数可以促进这样的事情,我会全力以赴。

即使任何人都只能用 Java 回答也没关系,因为我可以自己转换代码。如果我遗漏了一些明显的东西,我提前道歉。

Thi*_*iva 2

您可以执行类似于我在这个答案中所做的操作:/sf/answers/4557449361/

基本上,您将创建一个密封类作为您的属性inout. 您还需要将原始字符串包装成包含 String 的类型,这样您就可以使其扩展密封类,如下所示:

sealed class YourType {
    data class StringData(val value: String) : YourType()

    @JsonClass(generateAdapter = true)
    data class Record(
        val prop1: String,
        val prop2: Int
    ) : YourType()
}
Run Code Online (Sandbox Code Playgroud)

那么具有这些属性的模型将如下所示:

@JsonClass(generateAdapter = true)
data class Model(
    ...
    val in: YourType,
    val out: YourType,
    ...
)
Run Code Online (Sandbox Code Playgroud)

最后,您为该YourType类型编写自定义适配器:

class YourTypeCustomAdapter {
    @FromJson
    fun fromJson(jsonReader: JsonReader, delegate: JsonAdapter<Record>): YourType? {
        return if (jsonReader.peek() == BEGIN_OBJECT) {
            delegate.fromJson(jsonReader)
        } else {
            StringData(jsonReader.nextString())
        }
    }

    @ToJson
    fun toJson(jsonWriter: JsonWriter, yourType: YourType, delegate: JsonAdapter<Record>) {
        when (yourType) {
            is Record -> delegate.toJson(jsonWriter, yourType)
            is StringData -> jsonWriter.value(yourType.value)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

并在 Moshi 注册:

private val moshi = Moshi.Builder()
    .add(YourTypeCustomAdapter())
    .build()
Run Code Online (Sandbox Code Playgroud)