1

当我在 REPL 中执行时,它可以工作(由于mapWrites范围内的隐含):

scala> Map("a"->1l, "b"->2l)
res0: scala.collection.immutable.Map[String,Long] = Map(a -> 1, b -> 2)

scala> Map("c" -> res0, "d" -> res0)
res1:     scala.collection.immutable.Map[String,scala.collection.immutable.Map[String,Long]] = Map(c -> Map(a -> 1, b -> 2), d -> Map(a -> 1, b -> 2))

scala> import play.api.libs.json._
import play.api.libs.json._

scala> Json.toJson(res1)
res2: play.api.libs.json.JsValue = {"c":{"a":1,"b":2},"d":{"a":1,"b":2}}

为什么我的代码仍然无法编译(它与 REPL 中的类型相同)?

No Json deserializer found for type Map[String,Map[String,Long]]. Try to implement an implicit Writes or Format for this type.

[编辑] 我找到了一种解决方法,但我不明白我为什么需要它:

implicit def mapWrites = Writes[Map[String,Map[String,Long]]] ( m => Writes.mapWrites(Writes.mapWrites[Long]).writes(m))
4

1 回答 1

0

Play 2.1 JSON API 不为 Type 提供序列化程序Map[String, Object]

为特定类型定义case classandFormat而不是Map[String, Object]

// { "val1" : "xxx", "val2" : ["a", "b", "c"] }
case class Hoge(val1: String, val2: List[String])

implicit val hogeFormat = Json.format[Hoge]

如果您不想创建案例类。以下代码为 Map[String, Object] 提供 JSON 序列化器/反序列化器:

implicit val objectMapFormat = new Format[Map[String, Object]] {

  def writes(map: Map[String, Object]): JsValue =
    Json.obj(
      "val1" -> map("val1").asInstanceOf[String],
      "val2" -> map("val2").asInstanceOf[List[String]]
    )

  def reads(jv: JsValue): JsResult[Map[String, Object]] =
    JsSuccess(Map("val1" -> (jv \ "val1").as[String], "val2" -> (jv \ "val2").as[List[String]]))
}

Jackson 不知道如何反序列化 Scala 集合类scala.collection.immutable.Map,因为它没有实现任何 Java 集合接口。

您可以反序列化为 Java 集合:

val mapData = mapper.readValue(jsonContent, classOf[java.util.Map[String,String]])

或将 Scala 模块添加到映射器:

mapper.registerModule(DefaultScalaModule)
val mapData = mapper.readValue(jsonContent, classOf[Map[String,String]])

或者你也可以这样尝试

import play.api.libs.json._

    val a1=Map("val1"->"a", "val2"->"b")
    Json.toJSon(a1)

因为 a1 就Map[String,String]可以正常工作。

但是,如果我有更复杂的东西,比如我有的地方Map[String,Object],那就行不通了:

 val a = Map("val1" -> "xxx", "val2"-> List("a", "b", "c"))
    Json.toJSon(a1)
    >>> error: No Json deserializer found for type scala.collection.immutable.Map[String,Object]

我发现我可以执行以下操作:

val a2 = Map("val1" -> Json.toJson("a"), "val2" -> Json.toJson(List("a", "b", "c")))
Json.toJson(a2)

那行得通。

于 2013-11-14T10:48:23.057 回答