使用无形scala合并两个不同案例类的字段

azu*_*ras 7 scala shapeless

我想将两个不同案例类的字段合并为一个案例类.

例如,如果我有以下案例类:

case class Test(name:String, questions:List[Question], date:DateTime)

case class Answer(answers:List[Answers])
Run Code Online (Sandbox Code Playgroud)

我想要一个简洁的无形方式将两者合并为:

TestWithAnswers(name:String, questions:List[Question], date:DateTime, answers:List[Answer]). 
Run Code Online (Sandbox Code Playgroud)

那里有一个很好的无形答案吗?

Gab*_*lla 12

您可以使用无形Generic来执行此操作.

val t: Test = ???
val a: Answer = ???
val gt = Generic[Test]
val ga = Generic[Answer]
val gta = Generic[TestWithAnswers]

val ta = gta.from(gt.to(t) ++ ga.to(a))
Run Code Online (Sandbox Code Playgroud)

或者,如果你有一个单行的东西

val ta = Generic[TestWithAnswers].from(Generic[Test].to(t) ++ Generic[Answer].to(a))
Run Code Online (Sandbox Code Playgroud)

提供t了一个实例,Test并且a是一个实例Answer,ta将是一个实例TestWithAnswers.

快速解释:Generic可以将案例类转换为通用HList表示形式.因此,给定类的结构,您可以简单地将测试和答案转换为hlist,将两个列表连接成一个列表并TestWithAnswers从中构建实例.

这是一个更简单的示例,您可以复制粘贴并在REPL中尝试

import shapeless._
case class A(a: Int)
case class B(a: String)
case class AB(a: Int, b: String)
Generic[AB].from(Generic[A].to(A(42)) ++ Generic[B].to(B("foo")))
// AB(42, "foo")
Run Code Online (Sandbox Code Playgroud)