如何解析json并根据其内容提取到不同的case类

  • 本文关键字:提取 case json 何解析 scala json4s
  • 更新时间 :
  • 英文 :


我正在尝试使用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如何执行反序列化。在我们的案例中,我们需要专门化typeHintFieldNametypeHints。第一个是我们的类型提示的键,在本例中它是"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)
}

最新更新