2

我正在使用带有 playframework 2 的内置 jerson,我想要的只是序列化包含不同类型值的地图:

object AppWriter extends Writes[Application] {
    def writes(app: Application): JsValue = {
        Json.toJson(Map(
            "id" -> app.getId.toString,
            "name" -> app.getName,
            "users" -> Seq(1, 2, 3)
        ))
    }
}

在这种情况下,我有:

No Json deserializer found for type scala.collection.immutable.Map[java.lang.String,java.lang.Object]. 
Try to implement an implicit Writes or Format for this type

通过框架代码导航显示 Map[String,V] 类型隐式 def mapWrites[V] .. 的序列化程序,但我不明白为什么它不适用。

有谁能够帮助我?

UPD:我找到了简单的解决方法:

object AppWriter extends Writes[Application] {
def writes(app: Application): JsValue = {
    Json.toJson(Map[String, JsValue](
        "id" -> JsNumber(BigDecimal(app.getId)),
        "name" -> JsString(app.getName),
        "users" -> JsArray(Seq(1, 2, 3).map(x => JsNumber(x)))
    ))
}

}

但这不是那么优雅......

4

1 回答 1

1

执行此操作的标准方法是JsObject从字段的各个键值对创建一个,而不是将这些对放入映射中。例如,假设您Application看起来像这样:

case class Application(getId: Int, getName: String)

你可以写:

import play.api.libs.json._, Json.toJson

implicit object AppWriter extends Writes[Application] {
  def writes(app: Application): JsValue = JsObject(
    Seq(
      "id"    -> JsNumber(app.getId),
      "name"  -> JsString(app.getName),
      "users" -> toJson(Seq(1, 2, 3))
    )
  )
}

进而:

scala> toJson(Application(1, "test"))
res1: play.api.libs.json.JsValue = {"id":1,"name":"test","users":[1,2,3]}

请注意,您无需详细说明如何序列化Seq[Int]- 默认Format实例将为您完成工作。

于 2012-12-16T16:21:09.223 回答