我有一个数据类,其属性的类型是另一个数据类,如下所示:
@Serializable
data class Vehicle (
val color: String,
val miles: Int,
val year: Int,
val garage: Garage
)
@Serializable
data class Garage (
val latitude: Float,
val longitude: Float,
val name: String
)
在序列化时,它会产生如下输出:
{
"color" : "black" ,
"miles" : 35000 ,
"year" : 2017 ,
"garage" : { "latitude" : 43.478342 , "longitude" : -91.337000 , "name" : "Paul's Garage" }
}
但是,我希望garage
是其JSON表示的文字字符串,而不是实际的JSON对象。换句话说,期望的输出是:
{
"color" : "black" ,
"miles" : 35000 ,
"year" : 2017 ,
"garage" : "{ \"latitude\" : 43.478342 , \"longitude\" : -91.337000 , \"name\" : \"Paul's Garage\" }"
}
我如何在Kotlin中做到这一点?只用kotlinx.serialization
就可以做到吗?或者Jackson/Gson绝对有必要吗?
请注意,此输出用于特定的用法。我不能覆盖基本序列化程序,因为我仍然需要从普通JSON序列化/反序列化(第一个示例)。换句话说,最好的方案是将第一个JSON样本转换为第二个样本,而不必让数据类直接生成第二个样本。
谢谢!
发布于 2020-10-22 17:50:17
为Vehicle
创建自定义SerializationStrategy
,如下所示:
val vehicleStrategy = object : SerializationStrategy<Vehicle> {
override val descriptor: SerialDescriptor
get() = buildClassSerialDescriptor("Vehicle") {
element<String>("color")
element<Int>("miles")
element<Int>("year")
element<String>("garage")
}
override fun serialize(encoder: Encoder, value: Vehicle) {
encoder.encodeStructure(descriptor) {
encodeStringElement(descriptor, 0, value.color)
encodeIntElement(descriptor, 1, value.miles)
encodeIntElement(descriptor, 2, value.year)
encodeStringElement(descriptor, 3, Json.encodeToString(value.garage))
}
}
}
然后将其传递给Json.encodeToString()
val string = Json.encodeToString(vehicleStrategy, vehicle)
结果:
{"color":"black","miles":35000,"year":2017,"garage":"{\"latitude\":43.47834,\"longitude\":-91.337,\"name\":\"Paul's Garage\"}"}
更多信息here
发布于 2021-09-01 05:02:47
下面是一个解决方案,其中包含一个用于Garage
的自定义序列化程序和一个用于Vehicle
的附加类。
Garage
到String
序列化程序:
object GarageToStringSerializer : KSerializer<Garage> {
override val descriptor = PrimitiveSerialDescriptor("GarageToString", PrimitiveKind.STRING)
override fun serialize(encoder: Encoder, value: Garage) = encoder.encodeString(Json.encodeToString(value))
override fun deserialize(decoder: Decoder): Garage = Json.decodeFromString(decoder.decodeString())
}
辅助类:
@Serializable
data class VehicleDto(
val color: String,
val miles: Int,
val year: Int,
@Serializable(GarageToStringSerializer::class)
val garage: Garage
) {
constructor(v: Vehicle) : this(v.color, v.miles, v.year, v.garage)
}
可以通过以下方式接收所需的结果:
Json.encodeToString(VehicleDto(vehicle))
https://stackoverflow.com/questions/64478425
复制相似问题