编辑:
这是我的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()
答案 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 -> ""
}
}
}