我无法将org.joda.time.DateTime字段从JSON反序列化为case类。
JSON:
val ajson=parse(""" { "creationDate": "2013-01-02T10:48:41.000-05:00" }""")
我还设置了这些序列化选项:
implicit val formats = Serialization.formats(NoTypeHints) ++ net.liftweb.json.ext.JodaTimeSerializers.all
反序列化:
val val1=ajson.extract[Post]
帖子是:
case class Post(
creationDate : DateTime){ ... }
我得到的例外是:
net.liftweb.json.MappingException: No usable value for creationDate
Invalid date format 2013-01-02T10:48:41.000-05:00
如何将该日期字符串反序列化为DateTime对象?
编辑:
这有效:val date3= new DateTime("2013-01-05T06:24:53.000-05:00")
它使用与反序列化中的JSON相同的日期字符串。这里发生了什么?
答案 0 :(得分:9)
似乎它是Lift默认使用的DateParser
格式。在深入研究the code时,您可以看到解析器在将结果传递给DateParser.parse(s, format)
的构造函数之前尝试使用org.joda.time.DateTime
。
object DateParser {
def parse(s: String, format: Formats) =
format.dateFormat.parse(s).map(_.getTime).getOrElse(throw new MappingException("Invalid date format " + s))
}
case object DateTimeSerializer extends CustomSerializer[DateTime](format => (
{
case JString(s) => new DateTime(DateParser.parse(s, format))
case JNull => null
},
{
case d: DateTime => JString(format.dateFormat.format(d.toDate))
}
))
Lift似乎使用的格式为:yyyy-MM-dd'T'HH:mm:ss.SSS'Z'
要解决这个问题,您可以指定正确的模式并将其添加到序列化选项中,或者如果您希望让JodaTime构造函数完成所有工作,您可以创建自己的序列化程序,如:
case object MyDateTimeSerializer extends CustomSerializer[DateTime](format => (
{
case JString(s) => tryo(new DateTime(s)).openOr(throw new MappingException("Invalid date format " + s))
case JNull => null
},
{
case d: DateTime => JString(format.dateFormat.format(d.toDate))
}
))
然后将其添加到您的格式列表中,而不是net.liftweb.json.ext.JodaTimeSerializers.all
答案 1 :(得分:0)
可能不是100%优雅但只是几行,非常易读且有效:
val SourISODateTimeFormat = DateTimeFormat.forPattern("YYYY-MM-dd'T'HH:mm:ss.SSSZ")
val IntermediateDateTimeFormat = DateTimeFormat.forPattern("YYYY-MM-dd'T'HH:mm:ss'Z'")
def transformTimestamps(jvalue: JValue) = jvalue.transform {
case JField(name @ ("createdTime" | "updatedTime"), JString(value)) =>
val dt = SourceISODateTimeFormat.parseOption(value).get
JField(name, JString(IntermediateDateTimeFormat.print(dt)))
}