Android Kotlin - GSON JsonDeserializer 中的泛型来处理 Array vs Obj



我需要我们的 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 方法不起作用,因为我必须显式发送特定类型。

最新更新