首页 > 解决方案 > Kotlin:将嵌套的 JSON 对象转换为文字字符串

问题描述

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

@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 样本转换为第二个样本,而不一定让数据类直接生成第二个样本。

谢谢!

标签: jsonkotlinkotlinx.serialization

解决方案


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

更多信息在这里


推荐阅读