web-dev-qa-db-ja.com

JSONをタイプに変換する方法Scala

私の問題は、TwitterなどからJSONテキストを受信することです。次に、このテキストをscalaのネイティブオブジェクトに変換したいと思います。これを行うための標準的な方法はありますか? Play2も使用しています

これが私が持っているものです

import scala.io.Source.{fromInputStream}
import Java.net._

val url = new URL("https://api.Twitter.com/1/trends/1.json")
val content = fromInputStream( url.openStream ).getLines.mkString("\n")
val json = Json.parse(content)
val a = (json \ "trends" )
Ok(a(0))

JSONからすべてのトレンド名を取得したい

14

私は個人的にlift-jsonですが、これを行うのは非常に簡単です PlayのJSONライブラリ

import play.api.libs.json._
import scala.io.Source

case class Trend(name: String, url: String)

implicit object TrendReads extends Reads[Trend] {
  def reads(json: JsValue) = Trend(
    (json \ "name").as[String],
    (json \ "url").as[String]
  )
}

val url = new Java.net.URL("https://api.Twitter.com/1/trends/1.json")
val content = Source.fromInputStream(url.openStream).getLines.mkString("\n")
val trends = Json.parse(content) match {
  case JsArray(Seq(t)) => Some((t \ "trends").as[Seq[Trend]])
  case _ => None
}

現在、これにより次のようになります。

scala> trends.foreach(_.foreach(println))
Trend(#TrueFactsAboutMe,http://Twitter.com/search/?q=%23TrueFactsAboutMe)
Trend(#200mFinal,http://Twitter.com/search/?q=%23200mFinal)
Trend(Jamaica 1,2,3,http://Twitter.com/search/?q=%22Jamaica%201,2,3%22)
Trend(#DontComeToMyHouse,http://Twitter.com/search/?q=%23DontComeToMyHouse)
Trend(Lauren Cheney,http://Twitter.com/search/?q=%22Lauren%20Cheney%22)
Trend(Silver & Bronze,http://Twitter.com/search/?q=%22Silver%20&%20Bronze%22)
Trend(Jammer Martina,http://Twitter.com/search/?q=%22Jammer%20Martina%22)
Trend(Japan 2-0,http://Twitter.com/search/?q=%22Japan%202-0%22)
Trend(Prata e Bronze,http://Twitter.com/search/?q=%22Prata%20e%20Bronze%22)
Trend(Final 200m,http://Twitter.com/search/?q=%22Final%20200m%22)

そうそう、ほぼ正しいように見えます。

5
Travis Brown

Lift-Json をご覧ください。これはLiftWebフレームワークの一部ですが、スタンドアロンライブラリとして使用できます。 jsonをケースクラス(およびリストやマップなどのそれらのコレクション)に解析でき、アノテーションを追加する必要はありません。また、クラスをjsonとしてレンダリングし、jsonのマージとクエリをサポートします。

これは彼らのウェブサイトから取られた例です:

import net.liftweb.json._
implicit val formats = DefaultFormats // Brings in default date formats etc.

case class Child(name: String, age: Int,
                birthdate: Option[Java.util.Date])
case class Address(street: String, city: String)
case class Person(name: String, address: Address,
                 children: List[Child])
val json = parse("""
         { "name": "joe",
           "address": {
             "street": "Bulevard",
             "city": "Helsinki"
           },
           "children": [
             {
               "name": "Mary",
               "age": 5
               "birthdate": "2004-09-04T18:06:22Z"
             },
             {
               "name": "Mazy",
               "age": 3
             }
           ]
         }
       """)

json.extract[Person] 
/* Person = Person(joe, Address(Bulevard,Helsinki),
                  List(Child(Mary,5,Some(Sat Sep 04 18:06:22 EEST 2004)), 
                       Child(Mazy,3,None)))
 */
6

Jackson JSONプロセッサ を使用することをお勧めします。 JavaとScalaの両方で機能します。JSONデータをネイティブオブジェクトにマッピングする方法を説明するアノテーションをクラスに追加するだけです。

例:

import scala.reflect.BeanProperty
import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.annotate._

class User {
  @BeanProperty var gender: String = null
  @BeanProperty var verified: Boolean = false
  @BeanProperty var userImage: Array[Byte] = null
  @BeanProperty var name: Name = null
}

case class Name {
  @BeanProperty var first: String = null;
  @BeanProperty var last: String = null;
}

object Json {
  def main(argv: Array[String]) {
    val input = """{
  "name" : { "first" : "Joe", "last" : "Sixpack" },
  "verified" : false,
  "userImage" : "Rm9vYmFyIQ=="
}""";

    val mapper = new ObjectMapper(); // can reuse, share globally
    val user: User = mapper.readValue(input, classOf[User]);

    print(user.name.first);
  }
}

このソリューションには、すべてのフィールドに@BeanPropertyの注釈を付ける必要があるという少し面倒な作業がありますが、もっと簡単な方法はわかりません。


備考:私が知る限り、Jacksonはjavax.bean.Introspectorを使用せず、メソッド自体を調べてゲッター/セッターを見つけようとします。もしそうなら、物事はもっと簡単だっただろう、ただ書くことは可能だろう

@scala.reflect.BeanInfo
case class Name {
    var first: String;
    var last: String;
}
3
Petr Pudlák

Jerkson libを試してください: https://github.com/codahale/jerkson/

これは、scala Jacksonに基づくjsonライブラリです。2を再生するために含まれています: http://www.playframework.org/documentation/2.0.2/ScalaJson ==

例:

    case class Person(id: Long, name: String)
    parse[Person]("""{"id":1,"name":"Coda"}""") //=> Person(1,"Coda")
3
user1411778

json.parse(string)を使用してjsvalueに変換し、演算子as [T]を追加して値を抽出します

0