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

时间:2014-11-10 10:24:38

标签: scala json4s

我正在尝试使用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"}

这样的json

我正在寻找一种方法来概括解析器函数,以便我可以使用它来解析从Event继承的所有类型的事件,然后对函数的返回值进行模式匹配。

1 个答案:

答案 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如何执行反序列化。在我们的情况下,我们需要专门化typeHintFieldNametypeHints。第一个是我们的类型提示的关键,在示例中它是"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)
}