找不到类型列表选项的Json反序列化器

时间:2018-03-23 08:59:00

标签: json scala playframework-2.0

如何使用play json OFormat宏来获取选项列表?

val text = """[{"name": "John", "age": 30}, null, {"name": "Steve", "age": 34}]"""
import play.api.libs.json.Json
case class Person(name: String, age: Int)
implicit val personFormat = Json.format[Person]
val data = Json.parse(text).validate[List[Option[Person]]]

// Error: No Json deserializer found for type List[Option[Person]]. Try to implement an implicit Reads or Format for this type.

我正在做如下的解决方法:

val data = Json.parse(text).as[Array[JsValue]].toList.map {
  case JsNull => None
  case x => Some(x.validate[Person].get)
}
println(data)

// List(Some(Person(John,30)), None, Some(Person(Steve,34)))

如果没有这种解决方法,只使用OFormat宏,我该如何实现相同的效果?

1 个答案:

答案 0 :(得分:0)

不确定是否可以直接使用,但例如可以这样做(使用this回答):

val text = """[{"name": "John", "age": 30}, null, {"name": "Steve", "age": 34}]"""

import play.api.libs.json._

implicit def optionFormat[T: Format]: Format[Option[T]] = new Format[Option[T]]{
  override def reads(json: JsValue): JsResult[Option[T]] = json.validateOpt[T]

  override def writes(o: Option[T]) = o match {
    case Some(t) ⇒ implicitly[Writes[T]].writes(t)
    case None ⇒ JsNull
  }
}

case class Person(name: String, age: Int)

implicit val personFormat= {
  implicit val f = Json.format[Person]
  implicitly[Format[Option[Person]]]
}


val data = Json.parse(text).validate[List[Option[Person]]] // JsSuccess(List(Some(Person(John,30)), None, Some(Person(Steve,34))),)
Json.toJson(data.get) // [{"name":"John","age":30},null,{"name":"Steve","age":34}]

它无法通过自己的Reads[Person]翻译 - > Reads[Option[Person]] - > Reads[List[Option[Person]]]。我帮助Reads[Option[Person]]使用辅助方法。也许,模拟方法可以在游戏中使用..