我试图将来自REST API的响应建模为可以使用模式匹配的案例类。
我认为这是一个很好的适应方式,假设继承,但我看到这已经被弃用了。 我知道已经有与案例类和继承相关的问题,但我的问题更多地涉及如何在此处“没有”继承的情况下正确地建模以下内容。
我从以下两个案例类开始,它们工作得很好:
case class Body(contentType: String, content: String)
case class Response(statusCode: Int, body: Body)
例如,一个REST调用会返回这样的内容:
Response(200, Body("application/json", """{ "foo": "bar" }"""))
我可以像这样进行模式匹配:
response match {
case Response(200, Body("application/json", json)) => println(json)
case Response(200, Body("text/xml", xml)) => println(xml)
case Response(_,_) => println("Something unexpected")
}
等等。这些都很好用。
我遇到的问题是:我想为这些case class提供辅助扩展,例如:
case class OK(body: Body) extends Response(200, body)
case class NotFound() extends Response(404, Body("text/plain", "Not Found"))
case class JSON(json: String) extends Body("application/json", json)
case class XML(xml: String) extends Body("text/xml", xml)
这样我就可以进行简单的模式匹配,例如:
response match {
case OK(JSON(json)) => println(json)
case OK(XML(xml)) => println(xml)
case NotFound() => println("Something is not there")
// And still drop down to this if necessary:
case Response(302, _) => println("It moved")
}
此外,这也能让我的REST代码直接使用并返回:
Response(code, Body(contentType, content))
建立动态响应更容易。
因此...
我可以通过以下方式编译它(带有弃用警告):
case class OK(override val body: Body) extends Response(200, body)
然而,这似乎在模式匹配中无效。
Response(200, Body("application/json", "")) match {
case OK(_) => ":-)"
case _ => ":-("
}
res0: java.lang.String = :-(
你有没有任何想法可以使这个工作?我愿意考虑不同的方法,但这是我试图找到实用案例类的尝试。