2014-08-31 40 views
2

我使用Facebook的圖形API的陣列和響應類似於此:[json4s]:提取不同對象

{ 
    "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" 
     } 
     ] 
} 

我通過

val extracted = (json \ "data" \"from").extract[PostFrom] 
設法例如提取 from

但是我擔心,如果我使用這種技術,我需要多次傳遞Json來提取所需的所有值,這可能會導致性能不佳。

我究竟能夠將這些字段從非相似對象數組中提取出來的case類?

我試着用以下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

+1

「這可能會導致糟糕的表現」 - 直到你需要你不應該太在意性能。即如果它可以迭代通過json一百毫秒一毫秒,它可能不會影響太多,如果它遍歷它一次或四次 – Exupery 2014-08-31 13:31:40

+0

一般來說,我會同意,但我覺得更像是我沒有完全掌握如何解析那些更復雜的json結構與json4s,一般來說會很安靜。所以我希望有人能給我一個更好的解決方案 – Aranir 2014-08-31 16:34:03

+0

FWIW做json \「data」\「from」這樣的事情基本上是一個地圖查找。在你必須做1000次之前,這個懲罰並不高。對性能產生負面影響的是每個級別的元素數量。這是非常小的。所以是的,這不是固定的時間查詢,而是每個級別的元素數量,問題是否重要? – 2014-09-05 21:01:46

回答

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 
}