我正在开发一个Play应用程序,我正在尝试将Joda DateTime对象用于我的案例类。
package model
import org.joda.time.DateTime
import play.api.libs.json._
case class User(name: String, created: DateTime)
object User {
implicit val yourJodaDateReads = Reads.jodaDateReads("yyyy-MM-dd'T'HH:mm:ss.SSSZ")
implicit val yourJodaDateWrites = Writes.jodaDateWrites("yyyy-MM-dd'T'HH:mm:ss.SSSZ'")
implicit val userFormat = Json.format[User]
def main(args: Array[String]) {
val value = Json.parse("{ \"name\" : \"hello\" , \"created\" : \"2015-07-16T20:32:04.046+02:00\" }")
println(Json.toJson(new User("user", new DateTime())))
println(Json.fromJson(value))
}
}
基于此solution,我收到此错误:
Error:(18, -1) Play 2 Compiler:
/activator-1.3.2/notifier-app/app/model/Test.scala:18: ambiguous implicit values:
both value yourJodaDateReads in object User of type => play.api.libs.json.Reads[org.joda.time.DateTime]
and value userFormat in object User of type => play.api.libs.json.OFormat[model.User]
我正在使用Activator 1.3.2和Play 2.3.8。
你能告诉我一下吗?
提前致谢。
更新
我理解 play.api.libs.json.Reads
中的隐含值存在冲突implicit val DefaultJodaDateReads = jodaDateReads("yyyy-MM-dd")
如何解决此问题?
答案 0 :(得分:22)
期待更好的选择,在这里我的解决方法:
val dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SSSZ"
val jodaDateReads = Reads[DateTime](js =>
js.validate[String].map[DateTime](dtString =>
DateTime.parse(dtString, DateTimeFormat.forPattern(dateFormat))
)
)
val jodaDateWrites: Writes[DateTime] = new Writes[DateTime] {
def writes(d: DateTime): JsValue = JsString(d.toString())
}
val userReads: Reads[User] = (
(JsPath \ "name").read[String] and
(JsPath \ "created").read[DateTime](jodaDateReads)
)(User.apply _)
val userWrites: Writes[User] = (
(JsPath \ "name").write[String] and
(JsPath \ "created").write[DateTime](jodaDateWrites)
)(unlift(User.unapply))
implicit val userFormat: Format[User] = Format(userReads, userWrites)
答案 1 :(得分:19)
在游戏2.6中,序列化/反序列化joda DateTime json的规范方法是使用play-json-joda库。通过更新您的build.sbt
来Import the library。然后像这样创建json reader和json writer:
import play.api.libs.json.JodaWrites
implicit val dateTimeWriter: Writes[DateTime] = JodaWrites.jodaDateWrites("dd/MM/yyyy HH:mm:ss")
import play.api.libs.json.JodaReads
implicit val dateTimeJsReader = JodaReads.jodaDateReads("yyyyMMddHHmmss")
答案 2 :(得分:4)
我知道这个问题已经回答了一段时间,但我找到了一个更简洁的答案
val pattern = "yyyy-MM-dd'T'HH:mm:ss.SSSZ"
implicit val dateFormat = Format[DateTime](Reads.jodaDateReads(pattern), Writes.jodaDateWrites(pattern))
implicit val userFormat = Json.format[User]
答案 3 :(得分:3)
试试这些:
implicit val dateWrites = jodaDateWrites("yyyy-MM-dd'T'HH:mm:ss.SSSZ")
implicit val dateReads = jodaDateReads("yyyy-MM-dd'T'HH:mm:ss.SSSZ")
答案 4 :(得分:1)
我认为您应该在User
和Json.toJson
函数中设置Json.fromJson
类型。而不是
println(Json.toJson(new User("user", new DateTime())))
println(Json.fromJson(value))
尝试:
println(Json.toJson[User](new User("user", new DateTime())))
println(Json.fromJson[User](value))
当你明确设置类型时,框架会知道要使用什么读/写。
<强>更新强>
它不一定要为Json.toJson
函数设置类型,因为您将User
对象作为函数参数传递,框架在运行时确定类型。
但对于Json.fromJson[User]
,您必须设置类型,否则框架不知道您要读取的对象的类型。