1

我有一个数据类,它有一个类型是另一个数据类的属性,如下所示:

@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.serializationJackson/Gson 完成还是绝对必要?

请注意,此输出用于特定用途。我无法覆盖基本序列化程序,因为我仍然需要从普通 JSON 序列化/反序列化(第一个示例)。换句话说,最好的方案是将第一个 JSON 样本转换为第二个样本,而不一定让数据类直接生成第二个样本。

谢谢!

4

2 回答 2

1

SerializationStrategyVehicle如下创建一个自定义:

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\"}"}

更多信息在这里

于 2020-10-22T09:50:17.220 回答
0

这是一个带有自定义序列化程序Garage和附加类的解决方案Vehicle

GarageString序列化器:

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))
于 2021-09-01T05:02:47.437 回答