我正在尝试使用json4s以json格式解析不同类型的事件。我编写了一些案例类来表示从基类Event
继承的不同事件:
abstract class Event{ def EventType : String }
case class StartSession(val EventType: String, val Platform: String) extends Event
case class AdView(val EventType: String, val EventSubtype: String) extends Event
这是我用来解析StartSession
事件的函数:
def parser(json: String): Event = {
val parsedJson = parse(json)
val s = parsedJson.extract[StartSession]
return s
}
此函数将正确解析像{"EventType":"StartSession","Platform":"Portal"}
我正在寻找一种方法来概括解析器函数,以便我可以使用它来解析从Event
继承的所有类型的事件,然后对函数的返回值进行模式匹配。
答案 0 :(得分:1)
类型提示提供问题的解决方案。如果您有多态类型,例如Event
,则类型提示(此处为EventType
)告诉json4s应该对json对象的实际类型进行反序列化。
供参考,请查看github page of json4s。有一个名为“序列化多态列表”的部分。
由于类型提示仅用于反序列化,我们可以删除EventType
中的字段Event
。
abstract class Event
case class StartSession(Platform: String) extends Event
case class AdView(EventSubtype: String) extends Event
我们需要将Formats
个实例纳入extract
的范围。该实例告诉json4s如何执行反序列化。在我们的情况下,我们需要专门化typeHintFieldName
和typeHints
。第一个是我们的类型提示的关键,在示例中它是"EventType"
。后者是从字符串值到类的映射。如果我们使用类名作为值,那么ShortTypeHints
就可以了。 Othwerise,我们可以实现我们自己的专业TypeHints
。
具体解决方案如下:
def main(args: Array[String]): Unit ={
val json =
s"""
|[
| {
| "EventType": "StartSession",
| "Platform": "Portal"
| },
| {
| "EventType": "AdView",
| "EventSubtype": "SpecializedView"
| }
|]
""".stripMargin
implicit val formats = new DefaultFormats {
override val typeHintFieldName: String = "EventType"
override val typeHints: TypeHints =
ShortTypeHints(
List(classOf[StartSession], classOf[AdView])
)
}
val parsedJson = parse(json)
val s = parsedJson.extract[List[Event]]
println(s)
}