Gson或Moshi:POJO中的字段可以有两种类型,如何保存到任何字段

时间:2017-10-13 08:36:17

标签: json kotlin gson retrofit moshi

编辑:

这是我的json字符串:

json#1
{
    [
        {
            field1 : ""
            field2 : 0
            field3 : "Amount not fixed" or field : 250 // this field can be string or int
        },
        {
            field1 : ""
            field2 : 0
            field3 : "Amount not fixed" or field : 250 // this field can be string or int
        }

    ]
}

json#2
{
    field1 : ""
    field2 : 0
    field3 : "Amount not fixed" or field : 250 // this field can be string or int
}

或者它可以来自服务器的任何json字符串。这里的要点是可能有一个或多个可能具有动态值的字段(在这种情况下,field3可以是字符串或int)

然后我想将它们反序列化为任何POJO

class Temp1 {
    // field1 here
    // field2 here

    @SerializedName("field3")
    val field3Int: Int? = null

    @SerializedName("field3")
    val field3String: String? = null

}

这意味着如果从服务器发送的值是Int,我想将值设置为field3Int。如果是String,则设置为field3String

可能有其他POJO会有这些可能具有动态价值的字段。

感谢Serj的回答,但在编辑问题以显示我的实际情况后,我仍然无法在TypeAdapter类上工作。

顺便说一下。我和Retrofit2这样使用它:

val moshi = Moshi.Builder()
                    .add(MultitypeJsonAdapterAdapter())
                    .build()
            return Retrofit.Builder().baseUrl(baseUrl)

                    .addConverterFactory(MoshiConverterFactory.create(moshi))
                    .client(httpClient.build())
                    .build()

2 个答案:

答案 0 :(得分:4)

使用Moshi,您可以利用多态反序列化功能。只需编写一个将使用JsonReader#readJsonValue()的自定义适配器。请参阅以下代码:

data class Multitype constructor(val fieldInt: Int?, val fieldString: String?) {
  constructor(fieldInt: Int) : this(fieldInt, null)
  constructor(fieldString: String) : this(null, fieldString)
}

class MultitypeJsonAdapterAdapter {
   @FromJson fun fromJson(reader: JsonReader): Multitype {
      val jsonValue = reader.readJsonValue() as Map<String, Any?>
      val field = jsonValue["field"]
      return when (field) {
        is String -> Multitype(field)
        is Double -> Multitype(field.toInt()) // readJsonValue parses numbers as Double
        else -> throw JsonDataException("Expected a field of type Int or String")
      }
   }

   @ToJson fun toJson(writer: JsonWriter, value: Multitype?) {
     TODO("not implemented")
   }

}

class MultitypeJsonAdapterAdapterTest {
  @Test fun check() {
    val moshi = Moshi.Builder()
      .add(MultitypeJsonAdapterAdapter())
      .add(KotlinJsonAdapterFactory())
      .build()

    val adapter = moshi.adapter(Multitype::class.java)

    val fromJson1 = adapter.fromJson("""{ "field": 42 }""")
    assertThat(fromJson1).isEqualTo(Multitype(42))

    val fromJson2 = adapter.fromJson("""{ "field": "test" }""")
    assertThat(fromJson2).isEqualTo(Multitype("test"))
  }
}

答案 1 :(得分:1)

我想我得到了我想要达到的目标。而且不需要使用任何适配器。 如果某个字段可以包含任何动态类型,则需要在POJO中将其声明为Any。然后,如果你想使用它的实际值,你只需要检查它的类型并进行转换。所以POJO应该这样:

class Temp1 {
    // field1 here
    // field2 here

    @SerializedName("field3")
    val field3: Any? = null

    fun getField3Str() : String {
        return when (field3) {
            is String -> field3 as String
            is Int -> {
                "%d".format(field3 as Int)
            }
            else -> ""
        }
    }
}