我的问题是我收到来自Twitter的JSON文本。然后我想将此文本转换为scala中的本机对象。有没有标准的方法来做到这一点?我也在使用Play 2
这就是我所拥有的
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中获取所有趋势名称
答案 0 :(得分:5)
看看Lift-Json。它是Lift Web框架的一部分,但可以用作独立库。它可以将json解析为case类(以及它们的集合,例如列表和映射),并且它不需要你添加注释。它还支持将类渲染为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)))
*/
答案 1 :(得分:5)
我个人更喜欢lift-json
,但使用Play's JSON library执行此操作非常简单:
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)
所以是的,看起来是正确的。
答案 2 :(得分:3)
我建议使用Jackson JSON processor。它适用于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
注释每个字段,但我不知道更简单的方法。
备注:据我所知,杰克逊不使用javax.bean.Introspector
,它试图通过自己检查方法找到getter / setter。如果确实如此,事情本来会更容易,就可以写出
@scala.reflect.BeanInfo
case class Name {
var first: String;
var last: String;
}
答案 3 :(得分:3)
尝试使用Jerkson lib:https://github.com/codahale/jerkson/
它是基于Jackson的scala的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")
答案 4 :(得分:0)
通过使用Json.parse(string)转换为jsvalue,然后将运算符as [T]添加为提取值