Moshi + Retrofit-处理未知类型的JSON响应

时间:2020-06-03 22:09:00

标签: android json kotlin retrofit moshi

我正在使用Moshi,并且具有以下数据类

@JsonClass(generateAdapter = true)
data class A (
    @Json(name = "_id")
    val id: String?,
    @Json(name = "b")
    val b: B? = null
)

@JsonClass(generateAdapter = true)
data class B (
    @Json(name = "_id")
    val id: String?,
    @Json(name = "foo")
    val foo: String? = null
    @Json(name = "c")
    val c: C? = null
)

@JsonClass(generateAdapter = true)
data class C (
    @Json(name = "_id")
    val id: String?,
    @Json(name = "bar")
    val bar: String? = null
)

我的API有时将对象作为ID返回,而有时将其作为实际对象返回。例如,有时当我得到对象A时,它将返回

{
    _id: "111111111",
    b: {
        _id: "222222222",
        foo: "foo",
        c: {
            _id: "333333333",
            bar: "bar"
        }
    }
}

但其他时候它可能会返回

{
    _id: "111111111",
    b: "222222222"
}

{
    _id: "111111111",
    b: {
        _id: "222222222",
        foo: "foo",
        c: "333333333"
    }
}

如我们所见,它可能返回代表对象或填充的对象本身的字符串。如何创建自定义的Moshi适配器来处理此问题?如果它返回代表该对象的ID,我希望它创建一个仅填充了ID并将其余字段设置为null的对象。

我试图创建一个像这样的自定义适配器

class bAdapter {
    @FromJson
    fun fromJson(b: Any): B {
        return when (b) {
            is String -> B(b)
            else -> b as B
        }
    }
}

但是我得到了错误

com.squareup.moshi.JsonDataException: java.lang.ClassCastException: com.squareup.moshi.LinkedHashTreeMap cannot be cast to com.example.B

1 个答案:

答案 0 :(得分:1)

如果有人遇到相同的问题,我可以通过使用漂亮的@JsonQualifier注释来解决它,如下所示:

@Retention(AnnotationRetention.RUNTIME)
@JsonQualifier
internal annotation class ObjectString

class ObjectStringAdapter {
    @FromJson
    @ObjectString
    fun fromJson(id: String): Object {
        return Object(id)
    }

    @ToJson
    fun toJson(@ObjectString obj: Object): String {
        return obj.toString()
    }
}

class ObjectAdapter {
    @FromJson
    fun fromJson(jsonReader: JsonReader, @ObjectString stringAdapter: JsonAdapter<Object>, defaultAdapter: JsonAdapter<Object>): Object {
        return when (JsonReader.Token.STRING) {
            jsonReader.peek() -> {
                stringAdapter.fromJson(jsonReader)
            }
            else -> {
                defaultAdapter.fromJson(jsonReader)
            }
        }!!
    }
}

其中Object是您的班级。

基本上创建了两个适配器,一个用于字符串,一个用于对象,并使用jsonReader.peek()窥视该值并在运行时根据类型确定使用哪个。

以供参考:Moshi LocalDateTime adapter with multiple format