Как я могу сериализовать приведенные ниже данные? Это AppConfig, и значение value может быть любого типа, и я не могу использовать Any для сериализации.
[
{
"key": "PROFILE_PHOTO",
"value": "url"
},
{
"key": "RELATIONSHIP",
"value": 0.6
},
{
"key": "SEX",
"value": [ "Man", "Woman"]
},
{
"key": "IOS_VER",
"value": 5
}
]
@SimonJacobs Я использую kotlinx.serialization





Короткий ответ: с полиморфизмом. Нам нужны разные типы для представления разных типов объектов, которые можно получить.
В значительной степени выбор типов зависит от значения и использования объектов, а не просто от просмотра необработанных данных, как мы здесь.
Однако мы можем угадать подходящие типы на основе того, что вы дали.
Теперь мы хотим написать иерархию запечатанных классов для представления входящих объектов, потому что это упрощает полиморфную сериализацию ( документация):
@Serializable
@JsonClassDiscriminator("key")
sealed class ApiObject
@Serializable @SerialName("PROFILE_PHOTO")
data class ProfilePhoto(@SerialName("value") val url: String) : ApiObject()
@Serializable @SerialName("RELATIONSHIP")
data class Relationship(@SerialName("value") val coefficient: Double) : ApiObject()
@Serializable @SerialName("SEX")
data class Sex(@SerialName("value") val sexes: List<SexType>) : ApiObject()
@Serializable @SerialName("IOS_VER")
data class IosVersion(@SerialName("value") val versionNumber: Int) : ApiObject()
enum class SexType { Man, Woman }
Мы также использовали несколько функций библиотеки, чтобы создать более удобную иерархию классов. В частности:
Мы дали имя каждому возможному типу сообщения, которое можем получить от API, и использовали аннотацию SerialName, чтобы можно было присваивать осмысленные имена входящим полям «значений».
Поле «ключ» будет использоваться в качестве нашего «дискриминатора», который сообщает библиотеке, какой из классов выбрать. По умолчанию это поле «тип», поэтому нам нужно указать библиотеке использовать вместо него «ключ» с аннотацией JsonClassDiscriminator.
И, наконец, поскольку «ключевые» поля задаются в регистре кричащей змеи, нам необходимо скорректировать сериализованные имена классов с помощью аннотации SerialName, чтобы имена наших классов могли соответствовать обычной практике Kotlin.
Теперь мы готовы протестировать наш код:
val testMessage: String = """
[
{
"key": "PROFILE_PHOTO",
"value": "url"
},
{
"key": "RELATIONSHIP",
"value": 0.6
},
{
"key": "SEX",
"value": [ "Man", "Woman"]
},
{
"key": "IOS_VER",
"value": 5
}
]
"""
val deserializedMessage: List<ApiObject> = Json.decodeFromString(testMessage)
val expectedDeserializedMessage = listOf(
ProfilePhoto("url"),
Relationship(0.6),
Sex(listOf(SexType.Man, SexType.Woman)),
IosVersion(5)
)
println("Does our code work? ${expectedDeserializedMessage == deserializedMessage}")
Вам нужно указать, какую библиотеку сериализации вы используете.