3

我正在尝试将我的 Play 2.0.x 应用程序迁移到 Play 2.1-RC2 并偶然发现了以下问题。在我的应用程序中,我有一个看起来像这样的案例类:

case class Player(
  playerId: Pk[Long],
  name: Option[String],
  groupId: Long
)

在我的 Play 2.0.x 代码中,我有一个PlayerFormat对象用于读取和写入此类的 JSON 实例,如下所示:

object PlayerFormat extends Format[Player] {

def reads(json: JsValue): Player = Player(
    (json \ "id").asOpt[Long].map( Id(_) ).getOrElse( NotAssigned ),
    (json \ "name").asOpt[String],
    (json \ "group" \ "id").as[Long]
)

def writes(p: Player): JsValue = toJson(
  Map(
      "id" -> toJson(p.playerId.toOption),
      "name" -> toJson(p.name),
      "group" -> toJson(
          Map("id" -> p.groupId)
      )    
  )
)

}

问题是“我如何在 Play 2.1 中读取可选(可为空)属性“id”并根据它的存在将playerId属性设置为Id[Long]or NotAssigned

另外,这可以以某种方式重写以使用 JSON 初始宏吗?

4

1 回答 1

4

Json Macro Inception 在这里不是一个好的选择,因为这种情况太复杂了。它只支持经典案例,你不能举例map

在这种情况下,您需要一个特定的格式化程序

import play.api.libs.json._
import play.api.libs.functional.syntax._
import anorm._

implicit val playerFormat = (
    (__ \ "id").formatNullable[Long] and
    (__ \ "name").formatNullable[String] and
    (__ \ "group" \ "id").format[Long]
)((id, name, group) => Player(id.map(Id(_)).getOrElse(NotAssigned), name, group), 
  (p: Player) => (p.playerId.toOption, p.name, p.groupId))

这有点复杂,因为您的要求是;)

于 2013-01-15T20:53:47.907 回答