TypeAdapter для сериализации / десериализации объекта внутри класса модели - PullRequest
0 голосов
/ 26 сентября 2019

У меня есть ответ JSON, который выглядит следующим образом:

[{
    "id":"61051",
    "employee_name":"koiudshau",
    "employee_salary":"111",
    "employee_age":"0",
    "profile_image":""
},
{
    "id":"61071",
    "employee_name":"Nickky Raboy",
    "employee_salary":"1234",
    "employee_age":"20",
    "profile_image":""
}]

Я хотел создать пользовательский типизированный адаптер, который разделит значение employee_name на firstName и secondName и сохраните их как объект с именем Name, который содержит firstName и secondName в качестве параметров.Это должно быть сделано во время десериализации .

Во время сериализации объект типа Name должен преобразовываться в String (того же типа перед десериализацией).

Это коды, которые я 'мы пробовали.

Класс модели:

data class Employee (
    @SerializedName("id")
    val id: Int = 0,
    @SerializedName("employee_name")
    val employee_name: Name,
    @SerializedName("employee_salary")
    val employee_salary: Float = 0f,
    @SerializedName("employee_age")
    val employee_age: Int = 0,
    @SerializedName("profile_image")
    val profile_image : String = ""
)

data class Name(val firstName : String = "" , val secondName : String = "")

CustomizedTypeAdapterFactory:

abstract class CustomizedTypeAdapterFactory<C>(private val customizedClass: Class<C>) : TypeAdapterFactory {
    override fun <T> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
        return if (type.rawType == customizedClass) customizeMyClassAdapter(gson, type as TypeToken<C>) as TypeAdapter<T> else null
    }
    private fun customizeMyClassAdapter(gson: Gson, type: TypeToken<C>): TypeAdapter<C> {
        val delegate = gson.getDelegateAdapter(this, type)
        val elementAdapter = gson.getAdapter(JsonElement::class.java)
        return object : TypeAdapter<C>() {
            @Throws(IOException::class)
            override fun write(out: JsonWriter, value: C) {
                val tree = delegate.toJsonTree(value)
                beforeWrite(value, tree)
                elementAdapter.write(out, tree)
            }
            @Throws(IOException::class)
            override fun read(`in`: JsonReader): C {
                val tree = elementAdapter.read(`in`)
                afterRead(tree)
                return delegate.fromJsonTree(tree)
            }
        }
    }
    protected open fun beforeWrite(source: C, toSerialize: JsonElement) {}
    protected open fun afterRead(deserialized: JsonElement) {}
}

MyClassTypeAdapterFactory:

class MyClassTypeAdapterFactory : CustomizedTypeAdapterFactory<Name>(Name::class.java) {
    override fun beforeWrite(source: Name, toSerialize: JsonElement) {
        val custom = toSerialize.asJsonObject.get("employee_name").asJsonObject
        custom.add("size", JsonPrimitive(custom.entrySet().size))
    }
    override fun afterRead(deserialized: JsonElement) {
        val custom = deserialized.asJsonObject.get("employee_name").asJsonObject
        custom.remove("size")
    }
}

RetrofitBuilder:

private var retrofitBuilder: Retrofit.Builder = Retrofit.Builder().baseUrl(API_BASE_URL).addConverterFactory(createGsonConverter()).client(okHttpClientBuilder.build())

createGsonConverter ():

private fun createGsonConverter() : Converter.Factory {
    val gsonBuilder = GsonBuilder()
    gsonBuilder.registerTypeAdapterFactory(MyClassTypeAdapterFactory())
    val gson = gsonBuilder.create()
    return GsonConverterFactory.create(gson)
}

При запускеэтот код, приложение было сбой, показывая это сообщение журнала:

E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.shaheen.gsonparsing, PID: 15147
    com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: Not a JSON Object: "Karolina38606"

Ответы [ 2 ]

0 голосов
/ 26 сентября 2019

Я получил требуемый вывод, изменив класс CustomizedTypeAdapterFactory и метод createGsonConverter() следующим образом:

CustomizedTypeAdapterFactory:

class CustomizedTypeAdapterFactory : TypeAdapter<Name>() {
    override fun write(out: JsonWriter?, value: Name?) {
        out!!.value("${value?.firstName} ${value?.secondName}")
    }

    override fun read(`in`: JsonReader?): Name {
        var lastName = ""
        var firstName = ""
        val employeeName = `in`!!.nextString()
        if (employeeName!=null) {
            val name = employeeName.split(" ")
            if(name.size > 1){
                lastName += name[name.size - 1]
                for (element in 0 until (name.size - 1)){
                    firstName = if (element == (name.size - 1)) "$firstName${name[element]}" else "$firstName${name[element]} "
                }
            }
            else firstName = name[0]
        }
        return Name(firstName,lastName)
    }
}

createGsonConverter ():

private fun createGsonConverter() : Converter.Factory {
    val gsonBuilder = GsonBuilder()
    gsonBuilder.registerTypeAdapter(Name::class.java,CustomizedTypeAdapterFactory()).setLenient()
    val gson = gsonBuilder.create()
    return GsonConverterFactory.create(gson)
}
0 голосов
/ 26 сентября 2019

"employee_name" - это строка, не являющаяся jsonobject

...