3

注意:我正在学习无形,所以如果我错过任何细节,请要求澄清。

背景:

在练习 Shapeless 时,我正在为固定长度格式构建编码/解码解决方案。这个想法是每个case class都有自己的编码器/解码器,定义为HList与其属性对齐。

即使两个类共享相同的属性,它们的编码也可能不同。每个字段的描述将包含一些 (4) 值。但这在问题中并不重要。

问题:

完整代码可在此处获得:https ://github.com/atais/Fixed-Length/blob/03b395947a6b00e548ea1e76a9660e471f136565/src/main/scala/test/Main.scala

我声明了一个示例案例类及其编码器:

case class Employee(name: String, number: Int, manager: Boolean)

object Employee {
  implicit val employeeEncoder =
    FLEncoder.fixed((s: String) => s) ::
    FLEncoder.fixed((s: String) => s) ::
    FLEncoder.fixed((s: Int) => s.toString) ::
    FLEncoder.fixed((s: Boolean) => s.toString) ::
      HNil
}

所以我的类型employeeEncoder很漂亮:

::[FLEncoder[String], ::[FLEncoder[String], ::[FLEncoder[Int], ::[FLEncoder[Boolean], HNil]]]]

现在,编码器implicitly正在寻找一个FLEncoder[Employee],我希望它可以是上面的实现。

我已使用此解决方案将 TypeClasses 组合为元组:

但我得到:

Error:(69, 17) could not find implicit value for parameter enc: test.FLEncoder[test.Employee]
  println(encode(example))

如果我分别声明这些编码器,它们工作正常

implicit val a = fixed((s: String) => s)
implicit val b = fixed((s: Int) => s.toString)
implicit val c = fixed((s: Boolean) => s.toString)

问题:

所以基本上,如何使用Shapeless它会知道这Hlist是一个对齐的编码器类型case class


类似的问题在 scodec 中得到解决。如果您在此处查看演示: https ://github.com/atais/Fixed-Length/blob/03b395947a6b00e548ea1e76a9660e471f136565/src/main/scala/test/Fixed.scala

你可以做这样的转变:

case class Person(name: String, age: Int)

val pc: Codec[shapeless.::[String, shapeless.::[Int, HNil]]] = (("name" | fixed(10, '_')) :: ("age" | fixed(6, '0').narrow[Int](strToInt, _.toString)))

val personCodec: Codec[Person] = pc.as[Person]

但我不知道如何在我的情况下使用TransformSyntax.as

4

1 回答 1

2

您需要提供一种方法来告诉编译器您的列表FLEncoder可以转换为FLEncoder列表。由于我们处于类型级别,因此可以使用类型类来完成:

trait ListOfEncoder[L <: HList] {
  type Inside <: HList
  def merge(l: L): FLEncoder[Inside]
}

object ListOfEncoder {
  type Aux[L <: HList, I <: HList] = ListOfEncoder[L] { type Inside = I }

  implicit val hnil: Aux[HNil, HNil] = new ListOfEncoder[HNil] {
    type Inside = HNil
    def merge(l: HNil) = FLEncoder.fixed(_ => "")
  }

  implicit def hcons[H, T <: HList](implicit T: ListOfEncoder[T]): Aux[FLEncoder[H] :: T, H :: T.Inside] = new ListOfEncoder[FLEncoder[H] :: T] {
    type Inside = H :: T.Inside
    def merge(l: FLEncoder[H] :: T): FLEncoder[H :: T.Inside] =
      FLEncoder.fixed((ht: H :: T.Inside) => l.head.encode(ht.head) + T.merge(l.tail).encode(ht.tail))
  }
}

现在,在您的 `FLEncoder 对象中,添加这个隐式类:

implicit class EncoderAsGeneric[L <: HList, I <: HList](l: L)(implicit L: ListOfEncoder.Aux[L, I]) {
  def as[E](implicit gen: Generic.Aux[E, I]) = 
    FLEncoder.fixed((e: E) => L.merge(l).encode(gen.to(e))
}

这将允许您定义

implicit val employeeEncoder = (fixed((s: String) => s) ::
  fixed((s: String) => s) ::
  fixed((s: Int) => s.toString) ::
  fixed((s: Boolean) => s.toString) ::
    HNil).as[Employee]

现在,所有隐含的内容都应该在您的范围内Main

顺便说一句,由于您使用 定义了 FLEncoder HList,因此不再需要ProductTypeClassCompanion,因为这仅用于从基本情况进行推断。

于 2017-06-07T08:27:55.343 回答