[json4s]:提取不同对象的数组

Ara*_*nir 3 scala facebook-graph-api json4s

我使用的是Facebook图形API,响应看起来与此类似:

{
  "data": [
    {
      "id": "311620272349920_311718615673419", 
      "from": {
        "id": "1456046457993048", 
        "name": "Richard Ettinson"
      }, 
      "to": {
        "data": [
          {
            "id": "311620272349920", 
            "name": "Barbara Fallerman"
          }
        ]
      }, 
      "with_tags": {
        "data": [
          {
            "id": "311620272349920", 
            "name": "Barbara Fallerman"
          }
        ]
      }, 
      "message": "I was gong out with her", 
      "actions": [
        {
          "name": "Comment", 
          "link": "https://www.facebook.com/311620272349920/posts/311718615673419"
        }, 
        {
          "name": "Like", 
          "link": "https://www.facebook.com/311620272349920/posts/311718615673419"
        }
      ]
}
Run Code Online (Sandbox Code Playgroud)

我设法通过例如提取from字段

val extracted = (json \ "data" \"from").extract[PostFrom]
Run Code Online (Sandbox Code Playgroud)

但我担心如果我使用这种技术,我需要多次传递Json来提取我需要的所有值,这可能会导致性能不佳.

我怎样才能从非相似对象数组中将这些字段提取到案例类中?

我尝试了以下内容case classes:

abstract class BaseResponse()
case class Data(list:List[Post])
case class Post(id: String, post: PostFrom) extends BaseResponse
case class PostFrom(id: String, name:String)
Run Code Online (Sandbox Code Playgroud)

这总是导致一个空的List,有没有办法找回一个Data有我感兴趣的某些类列表的类?(例如顶级 id,fromwith_tags)

Ara*_*nir 5

我发现的一种可能性是使用更多的case类而不是继承:

case class Root[T](data:Option[T])
case class Post(id: String, from: From, message: String)
case class From(id: String, name:String)
Run Code Online (Sandbox Code Playgroud)

基本上必须有一个根对象,它采用某种图形响应对象,另外它是可选的,因此如果解析响应有问题,它不会抛出异常.

然后我用以下方式使用它:

val body = r.entity.asString
val json = parse(r.entity.asString)
val root = json.extract[Root[Post]]

root.data match {
  case Some(post) =>
      val tagger = Tagger(post.from.id, post.from.name, post.id, post.message)
      log.info(s"We received $tagger")
      originalSender ! RetrievedTagger(tagger)

  case None => originalSender ! NoTaggerFound
}
Run Code Online (Sandbox Code Playgroud)