0

我在 Play 2.1 Scala 中为某些 JSON 编写 Format 对象时遇到了一些麻烦。我的(大大简化的)JSON 看起来像这样:

{
  "id": "id-1",
  "data": {
    "foo": "teststr1",
    "bar": "teststr2"
  }
}

我想将其序列化/反序列化为稍微简单的扁平格式:

case class TestData(id: String, foo: String, bar: String)

我最初的尝试是这样的:

implicit val testCaseFormat: Format[TestData] = (
  (__ \ "id").format[String] and
  (__ \ "data" \ "foo").format[String] and
  (__ \ "data" \ "bar").format[String]
)(TestData.apply, unlift(TestData.unapply))

这在读取bar数据时工作正常,但在输出 JSON 中仅写入最后一个“数据”值 ( ) 时,可能是因为data每个后续值都覆盖了该对象:

{
   "id" : "id-1",
   "data":{
     "bar":"teststr2"
   }
 }

有没有正确的方法来进行这种扁平化和非扁平化,而不诉诸分离ReadsWrites对象?我的实际数据结构非常大(虽然不太复杂),所以简洁性很好。

4

1 回答 1

2

如果我理解正确,您可以/应该以常规方式创建您的格式:

(__ \ "id").format[String] and
(__ \ "data").format(
  (__ \ "foo").format[String] and
  (__ \ "bar").format[String]
  tupled
)

然后只是提供不同applyunapply方法。

def specialApply(data:(String, (String, String))) = {
  val(id, (foo, bar)) = data
  TestData(id, foo, bar)
}

def specialUnapply(data:TestData):Option[(String, (String, String))] =
  Option(data) map { testData =>
    (testData.id, (testData.foo, testData.bar))
  }

编辑

您可以通过组合 apply 和 unapply 方法来避免将方法添加到伴随对象:

implicit val testCaseFormat: Format[TestData] = (
    // ...
)(TestData.tupled.compose { data:(String, (String, String)) => 
    val (id, (foo, bar)) = data
    (id, foo, bar)
  }), unlift(TestData.unapply))
于 2013-02-16T11:24:45.497 回答