这个问题最近出现了几次,所以我在这里把它作为常见问题整理一下。假设我有一些像这样的案例类:
这个问题最近出现了几次,所以我在这里把它作为常见问题整理一下。假设我有一些像这样的案例类:
import io.circe._, io.circe.generic.semiauto._
object model {
case class A(a: String)
case class B(a: String, i: Int)
case class C(i: Int, b: Boolean)
implicit val encodeA: Encoder[A] = deriveEncoder
implicit val encodeB: Encoder[B] = deriveEncoder
implicit val encodeC: Encoder[C] = deriveEncoder
implicit val decodeA: Decoder[A] = deriveDecoder
implicit val decodeB: Decoder[B] = deriveDecoder
implicit val decodeC: Decoder[C] = deriveDecoder
}
我想使用circe和Shapeless coproducts将值编码为JSON,这个值可能是其中的任何一个。
import io.circe.shapes._, io.circe.syntax._
import shapeless._
import model._
type ABC = A :+: B :+: C :+: CNil
val c: ABC = Coproduct[ABC](C(123, false))
一开始看起来这很好:
scala> c.asJson
res0: io.circe.Json =
{
"i" : 123,
"b" : false
}
但问题在于,我永远无法解码包含B
元素的余积类型,因为任何可以被解码为B
的有效JSON文档也可以被解码为A
,而circe-shapes提供的余积类型解码器会按照在余积类型中出现的顺序尝试解码元素。
scala> val b: ABC = Coproduct[ABC](B("xyz", 123))
b: ABC = Inr(Inl(B(xyz,123)))
scala> val json = b.asJson
json: io.circe.Json =
{
"a" : "xyz",
"i" : 123
}
scala> io.circe.jawn.decode[ABC](json.noSpaces)
res1: Either[io.circe.Error,ABC] = Right(Inl(A(xyz)))
如何在我的编码中消除余积的元素歧义?