2

我尝试将我的 Web 应用程序从 Play 2.0.4 迁移到 Play 2.1-RC2。

我有一个带有未知键列表的 JSON 数据(key1, key2),如下所示:

{description: "Blah",
 tags: [
   key1: ["value1", "value2"],
   key2: ["value3"]
 ]
}

我想将 JSON 中的数据存储在元标记列表中。在 Play 2.0.4 中,我使用了类似的东西来读取tags-list:

def readMetatags(meta: JsObject): List[Metatag] =
  meta.keys.toList.map(x => Metatag(x, (meta \ x).as[List[String]])

现在我想使用新的 Play 2.1-JSON-API(原型):

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

object Metatags {  
  implicit val metatagsRead: Read[Metatags] = (
    (JsPath \ "description").read[String] and
    (JsPath \ "tags").read[List[Metatag]]
  )(Metatags.apply _, unlift(Metatags.unapply _))

  implicit val metatagRead: Read[Metatag] = (
    JsPath().key. ?? read[String] and              // ?!? read key
    JsPath().values. ?? read[List[String]]         // ?!? read value list
  )(Metatag.apply _, unlift(Metatag.unapply _))

}

case class Metatags(description: String, tags: List[Metatag])
case class Metatag(key: String, data: List[String])

如何从 JSON 中读取密钥?

4

1 回答 1

5

这是一个为MetaTag班级定制阅读器的解决方案。read 只是将 转换JsValue为 a JsObject,它具有有用的fieldSet方法。

因为MetaTags, 宏 inception 完美运行

object Application extends Controller {

  case class MetaTags(description: String, tags: List[MetaTag])
  case class MetaTag(key: String, data: List[String])

  implicit val readMetaTag = Reads(js => 
    JsSuccess(js.as[JsObject].fieldSet.map(tag => 
      MetaTag(tag._1, tag._2.as[List[String]])).toList))

  implicit val readMetaTags = Json.reads[MetaTags]

  def index = Action {
    val json = Json.obj(
      "description" -> "Hello world",
      "tags" -> Map(
        "key1" -> Seq("key1a", "key1b"),
        "key2" -> Seq("key2a"),
        "key3" -> Seq("Key3a", "key3b", "key3c")))

    val meta = json.as[MetaTags]
    Ok(meta.tags.map(_.data.mkString(",")).mkString("/"))
    // key1a,key1b/key2a/Key3a,key3b,key3c
  }

}
于 2013-01-14T19:48:40.097 回答