在不同类型的案例类之间安全地复制字段

时间:2014-04-21 07:17:13

标签: scala generic-programming case-class shapeless

假设您有类似以下的案例类

case class Test1(a:String,b:Int,c:Char)

case class Test2(a:String,b:Int)

并使用以下变量实例化类

val test1 = Test1("first",2,'3')

val test2 = Test2("1st",20)

有没有办法使用.copy方法(或其他方法)将Test2中的变量应用到Test1,比如

val test3 = test1.copy(test2) //Note this isn't valid scala code
// Result should be ("1st",20,'3')

如果纯scala中不可能这样做,那么在Shapeless 1/2中会怎么做(当前代码在Shapeless 1中,但我们计划在某个时间点升级到Shapeless 2)

1 个答案:

答案 0 :(得分:14)

在无形2.0.0中,这可以像这样完成,

scala> import shapeless._
import shapeless._

scala> case class Test1(a: String, b: Int, c: Char)
defined class Test1

scala> case class Test2(a: String, b: Int)
defined class Test2

scala> val test1 = Test1("first", 2, '3')
test1: Test1 = Test1(first,2,3)

scala> val test2 = Test2("1st", 20)
test2: Test2 = Test2(1st,20)

scala> val test1Gen = Generic[Test1]
test1Gen: ... = $1$$1@6aac621f

scala> val test2Gen = Generic[Test2]
test2Gen: ... = $1$$1@5aa4954c

scala> val test3 = test1Gen.from(test2Gen.to(test2) :+ test1.c)
test3: Test1 = Test1(1st,20,3)

请注意,这会假设每个案例类中的字段顺序,而不是使用字段标签信息。如果你有多个相同类型的字段,这可能是容易出错的:类型可能排成一行,但潜在的语义可能会改变。

我们可以通过使用无形的LabelledGeneric来解决这个问题。 LabelledGeneric将case类值映射到无形可扩展记录,这些记录除了捕获字段值的类型外,还通过相应Scala {1}}的单例类型对类型中的字段名称进行编码。通过一些额外的基础设施(我将很快将其添加到无形的2.1.0),这使我们可以安全地在案例类之间进行映射,最小的样板,

Symbol