枚举 ADT 构造函数
获得所需表示的最直接方法是对案例类使用泛型派生,但为 ADT 类型显式定义实例:
import cats.syntax.functor._
import io.circe.{ Decoder, Encoder }, io.circe.generic.auto._
import io.circe.syntax._
sealed trait Event
case class Foo(i: Int) extends Event
case class Bar(s: String) extends Event
case class Baz(c: Char) extends Event
case class Qux(values: List[String]) extends Event
object Event {
implicit val encodeEvent: Encoder[Event] = Encoder.instance {
case foo @ Foo(_) => foo.asJson
case bar @ Bar(_) => bar.asJson
case baz @ Baz(_) => baz.asJson
case qux @ Qux(_) => qux.asJson
}
implicit val decodeEvent: Decoder[Event] =
List[Decoder[Event]](
Decoder[Foo].widen,
Decoder[Bar].widen,
Decoder[Baz].widen,
Decoder[Qux].widen
).reduceLeft(_ or _)
}
请注意,我们必须在解码器上调用widen
(由 Cats 的Functor
语法提供,我们在第一次导入时将其引入范围),因为Decoder
类型类不是协变的。circe 类型类的不变性是一个有争议的问题(例如 Argonaut 已经从不变变为协变并返回),但它有足够的好处,它不太可能改变,这意味着我们偶尔需要这样的变通方法。
还值得注意的是,我们的显式Encoder
和Decoder
实例将优先于我们从导入中获得的通用派生实例(有关此优先级如何工作的一些讨论,io.circe.generic.auto._
请参见我的幻灯片)。
我们可以像这样使用这些实例:
scala> import io.circe.parser.decode
import io.circe.parser.decode
scala> decode[Event]("""{ "i": 1000 }""")
res0: Either[io.circe.Error,Event] = Right(Foo(1000))
scala> (Foo(100): Event).asJson.noSpaces
res1: String = {"i":100}
这很有效,如果您需要能够指定 ADT 构造函数的尝试顺序,这是目前最好的解决方案。但是,必须像这样枚举构造函数显然并不理想,即使我们免费获得了案例类实例。
更通用的解决方案
正如我在 Gitter 上所指出的,我们可以通过使用 circe-shapes 模块来避免写出所有案例的麻烦:
import io.circe.{ Decoder, Encoder }, io.circe.generic.auto._
import io.circe.shapes
import shapeless.{ Coproduct, Generic }
implicit def encodeAdtNoDiscr[A, Repr <: Coproduct](implicit
gen: Generic.Aux[A, Repr],
encodeRepr: Encoder[Repr]
): Encoder[A] = encodeRepr.contramap(gen.to)
implicit def decodeAdtNoDiscr[A, Repr <: Coproduct](implicit
gen: Generic.Aux[A, Repr],
decodeRepr: Decoder[Repr]
): Decoder[A] = decodeRepr.map(gen.from)
sealed trait Event
case class Foo(i: Int) extends Event
case class Bar(s: String) extends Event
case class Baz(c: Char) extends Event
case class Qux(values: List[String]) extends Event
接着:
scala> import io.circe.parser.decode, io.circe.syntax._
import io.circe.parser.decode
import io.circe.syntax._
scala> decode[Event]("""{ "i": 1000 }""")
res0: Either[io.circe.Error,Event] = Right(Foo(1000))
scala> (Foo(100): Event).asJson.noSpaces
res1: String = {"i":100}
这将适用于任何范围内encodeAdtNoDiscr
的任何 ADT。decodeAdtNoDiscr
如果我们希望它受到更多限制,我们可以A
在这些定义中用我们的 ADT 类型替换泛型,或者我们可以使定义为非隐式,并为我们希望以这种方式编码的 ADT 显式定义隐式实例。
这种方法的主要缺点(除了额外的 circe-shapes 依赖)是构造函数将按字母顺序尝试,如果我们有模棱两可的案例类(其中成员名称和类型相同),这可能不是我们想要的)。
未来
generic-extras 模块在这方面提供了更多的可配置性。我们可以这样写,例如:
import io.circe.generic.extras.auto._
import io.circe.generic.extras.Configuration
implicit val genDevConfig: Configuration =
Configuration.default.withDiscriminator("what_am_i")
sealed trait Event
case class Foo(i: Int) extends Event
case class Bar(s: String) extends Event
case class Baz(c: Char) extends Event
case class Qux(values: List[String]) extends Event
接着:
scala> import io.circe.parser.decode, io.circe.syntax._
import io.circe.parser.decode
import io.circe.syntax._
scala> (Foo(100): Event).asJson.noSpaces
res0: String = {"i":100,"what_am_i":"Foo"}
scala> decode[Event]("""{ "i": 1000, "what_am_i": "Foo" }""")
res1: Either[io.circe.Error,Event] = Right(Foo(1000))
我们有一个额外的字段来指示构造函数,而不是 JSON 中的包装器对象。这不是默认行为,因为它有一些奇怪的极端情况(例如,如果我们的一个案例类有一个名为 的成员what_am_i
),但在许多情况下它是合理的,并且自从该模块被引入以来它在 generic-extras 中得到支持。
这仍然没有得到我们想要的东西,但它比默认行为更接近。我也一直在考虑更改withDiscriminator
为采用 aOption[String]
而不是 a String
,None
表明我们不想要一个指示构造函数的额外字段,从而为我们提供与上一节中的 circe-shapes 实例相同的行为。
如果您有兴趣看到这种情况发生,请打开一个问题,或者(甚至更好)一个拉取请求。:)