私の問題は、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からすべてのトレンド名を取得したい
私は個人的に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)
そうそう、ほぼ正しいように見えます。
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)))
*/
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;
}
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")
json.parse(string)を使用してjsvalueに変換し、演算子as [T]を追加して値を抽出します