我正在尝试使用json4s以json格式解析不同类型的事件。我已经编写了一些case类来表示不同的事件,所有这些事件都继承自基类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"}
的json
我正在寻找一种方法来概括解析器函数,这样我就可以使用它来解析从Event
继承的所有类型的事件,然后对函数的返回值进行模式匹配。
类型提示提供了问题的解决方案。如果您有一个多态类型,比如Event
,那么类型提示(这里是EventType
)告诉json4s应该将json对象反序列化为哪种实际类型。
为了参考,请查看json4s的github页面。有一个部分名为"序列化多态列表"。
由于类型提示仅用于反序列化,因此我们可以去掉Event
中的字段EventType
。
abstract class Event
case class StartSession(Platform: String) extends Event
case class AdView(EventSubtype: String) extends Event
我们需要将一个Formats
实例纳入extract
的范围。实例告诉json4s如何执行反序列化。在我们的案例中,我们需要专门化typeHintFieldName
和typeHints
。第一个是我们的类型提示的键,在本例中它是"EventType"
。后者是从字符串值到类的映射。如果我们使用类名作为值,那么ShortTypeHints
就可以了。此外,我们还可以实现自己的专用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)
}