我正在尝试将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]
或NotAssigned
?”
另外,这可以以某种方式重写为使用JSON初始宏吗?
答案 0 :(得分: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))
根据您的要求,它有点复杂;)