簡體   English   中英

如何使用scala中的play-json從Json中提取額外的(鍵,值)?

[英]How to extract extra (key,value) from Json using play-json in scala?

我有以下情況:

case class Person(id: Int, name: String)
val json = Json.obj("id" -> 1, "name" -> "John", "address"-> "Paris", "contact" -> "1234")

在這里,我想從json中提取額外的(鍵,值) ,即不屬於Person的 {“address” - >“Paris”,“contact” - >“1234”}

到目前為止我已經開發了以下方法:

case class Person(id: Int, name: String)
  val personReads = Json.reads[Person]
  val personWrites = Json.writes[Person]
  val json = Json.obj("id" -> 1, "name" -> "John", "address"-> "Paris", "contact" -> "1234")

  val person: Person = personReads.reads(json).get

  // This person json does not have extra fields 
  val personJson: JsObject = personWrites.writes(person).asInstanceOf[JsObject]

  val extraKeys = json.keys.diff(personJson.keys)

  val extraJson = extraKeys.foldLeft(Json.obj()){(result,key) =>
                            result.+(key -> json.\(key).get)}

  // {"address":"Paris","contact":"1234"}

這有效但在這里我必須做很多json到case類的轉換。 在這種情況下提取額外(鍵,值)的最佳方法是什么?

如果你想讓它適用於任何案例類而不想使用自定義Reads做任何事情,你可以使用反射或無形來提取案例類名稱,然后從你想要解析的對象中刪除它們。

例如,使用反射,這只創建一個case類實例一次,根本不需要Writes

import play.api.libs.json._
import scala.reflect.runtime.universe._

def withExtra[A: Reads: TypeTag]: Reads[(A, JsObject)] = {
  val ccFieldNames = typeOf[A].members.collect {
    case m: MethodSymbol if m.isCaseAccessor => m.name.toString
    }.toVector

  for {
    jsObj <- implicitly[Reads[JsObject]]
    a <- implicitly[Reads[A]]
    filteredObj = ccFieldNames.foldLeft(jsObj)(_ - _)
  } yield (a, filteredObj)
}

並使用它,例如:

case class Person(id: Int, name: String)
case class Location(id: Int, address: String)

val json = Json.obj("id" -> 1, "name" -> "John", "address"-> "Paris", "contact" -> "1234")

implicit val pReads = Json.reads[Person]
implicit val lReads = Json.reads[Location]

assert { withExtra[Person].reads(json).get == (
  Person(1, "John"),
  Json.obj("address"-> "Paris", "contact" -> "1234")
) }

assert { withExtra[Location].reads(json).get == (
  Location(1, "Paris"),
  Json.obj("name" -> "John", "contact" -> "1234")
) }

那里有可運行的代碼

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM