3

我正在使用 facebook graph 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"
        }
      ]
}

例如,我设法from通过

val extracted = (json \ "data" \"from").extract[PostFrom]

但我担心,如果我使用这种技术,我将需要多次传递 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)

这总是导致一个空列表,有没有办法找回一个Data包含我感兴趣的某些类的列表的类?(例如顶层 idfromwith_tags

4

1 回答 1

5

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

case class Root[T](data:Option[T])
case class Post(id: String, from: From, message: String)
case class From(id: String, name:String)

基本上必须有一个根对象,它接受某种图形响应对象,另外它是可选的,这样如果响应的解析出现问题,它就不会抛出异常。

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

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
}
于 2014-09-05T11:25:39.080 回答