我需要我们的 GSON 解析器能够处理数组与单个对象。 我已经让它工作到它可以识别任何 MutableList 并落入下面的反序列化器适配器的位置。 但是,我缺少一块在实际序列化为对象时不喜欢"T"类型的部分。 我知道它此时需要该类型,但是鉴于我们这里的内容,我是否缺少一些能够推断这一点的东西?
class JsonArrayObjectAdapter<T>: JsonDeserializer<MutableList<T>> {
override fun deserialize(json: JsonElement?, typeOfT: Type?, context: JsonDeserializationContext?): MutableList<T> {
val list = mutableListOf<T>()
json?.let {
context?.let {
if (json.isJsonArray) {
for (element in json.asJsonArray) {
val obj: T = context.deserialize(element, T::class.java) // does not like this
list.add(obj)
}
} else if (json.isJsonObject) {
val obj: T = context.deserialize(json, T!!::class.java) // does not like this
list.add(obj)
}
}
}
return list
}
}
尝试使用T::class.java
时出现此编译错误:
Cannot use 'T' as reified type parameter. Use a class instead.
下面是我正在使用的数据类的示例,供参考:
data class ExampleDataClass(
@SerializedName("ExampleKey") val exampleVar: MutableList<ExampleSubDataClass>
)
以下是我如何创建gson:
val listType = object : TypeToken<MutableList<ExampleDataSubClass>>() {}.getType()
val gson = GsonBuilder()
.registerTypeAdapter(listType, JsonArrayObjectAdapter<ExampleDataSubClass>())
.create()
result = gson.fromJson(jsonString, ExampleDataClass::class.java)
为了解决这个问题,我不得不为每种类型的对象制作多个适配器,这些对象可以是数组或对象。 不幸的是,因为我有大约 50 种不同类型的对象,因此在 GsonBuilder 中调用"registerTypeAdapters"需要大量重复代码。
为了简化一点,我创建了一个基于给定 JsonElement 和泛型 T 对象的数组的通用方法:
fun <T> getJSONArray(json: JsonElement, type: Type, ctx:
JsonDeserializationContext): ArrayList<T> {
val list = ArrayList<T>()
if (json.isJsonArray) {
for (e in json.asJsonArray) {
list.add(ctx.deserialize<Any>(e, type) as T)
}
} else if (json.isJsonObject) {
list.add(ctx.deserialize<Any>(json, type) as T)
} else {
throw RuntimeException("Unexpected JSON type: " + json.javaClass)
}
return list
}
以及使用上述方法定义的适配器的示例:
class JsonArrayObjectAdapterExampleDataClass: JsonDeserializer<ArrayList<ExampleDataClass>> {
override fun deserialize(json: JsonElement, typeOfT: Type, ctx: JsonDeserializationContext): ArrayList<ExampleDataClass> {
return getJSONArray(json, ExampleDataClass::class.java, ctx)
}
}
简单地将 typeOfT
var 传递给 getJSONArray
方法不起作用,因为我必须显式发送特定类型。