scala – 在不同类型的案例类之间安全地复制字段
发布时间:2020-12-16 09:26:26 所属栏目:安全 来源:网络整理
导读:假设您有类似以下的案例类 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中的变量应用于Te
|
假设您有类似以下的案例类
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) 解决方法
在无形的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",'3')
test1: Test1 = Test1(first,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,3)
请注意,这会假设每个案例类中的字段顺序,而不是使用字段标签信息.如果您有多个相同类型的字段,这可能会出错:类型可能会排成一行,但潜在的语义可能会被更改. 我们可以通过使用无形的LabelledGeneric来解决这个问题. LabelledGeneric将case类值映射到无形可扩展记录,这些记录除了捕获字段值的类型外,还通过相应Scala符号的单例类型对类型中的字段名称进行编码.通过一些额外的基础设施(我将很快添加到无形的2.1.0),这使我们能够安全地在案例类之间进行映射,最小的样板, import shapeless._,record._,syntax.singleton._,ops.hlist.Remove
/**
* This will be in shapeless 2.1.0 ...
*
* Permute the elements of the supplied `HList` of type `L` into the same order
* as the elements of the `HList` of type `M`.
*/
trait Align[L <: HList,M <: HList] extends (L => M) {
def apply(l: L): M
}
object Align {
def apply[L <: HList,M <: HList]
(implicit alm: Align[L,M]): Align[L,M] = alm
implicit val hnilAlign: Align[HNil,HNil] = new Align[HNil,HNil] {
def apply(l: HNil): HNil = l
}
implicit def hlistAlign[L <: HList,MH,MT <: HList,R <: HList]
(implicit
select: Remove.Aux[L,(MH,R)],alignTail: Align[R,MT]): Align[L,MH :: MT] = new Align[L,MH :: MT] {
def apply(l: L): MH :: MT = {
val (h,t) = l.removeElem[MH]
h :: alignTail(t)
}
}
}
/**
* This,or something like it,will be in shapeless 2.1.0 ...
*
* Utility trait intended for inferring a field type from a sample value and
* unpacking it into its key and value types.
*/
trait Field {
type K
type V
type F = FieldType[K,V]
}
object Field {
def apply[K0,V0](sample: FieldType[K0,V0]) =
new Field { type K = K0; type V = V0 }
}
object OldWineNewBottle {
case class From(s1: String,s2: String)
case class To(s2: String,i: Int,s1: String)
val from = From("foo","bar")
val fromGen = LabelledGeneric[From]
val toGen = LabelledGeneric[To]
// Define the type of the i field by example
val iField = Field('i ->> 0)
val align = Align[iField.F :: fromGen.Repr,toGen.Repr]
// extend the fields of From with a field for 'i',permute into
// the correct order for To and create a new instance ...
val to = toGen.from(align('i ->> 23 :: fromGen.to(from)))
assert(to == To("bar",23,"foo"))
}
(编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
相关内容
- 好好的代码突然编译不过? 退出shell后,再次打开shell又编
- twitter-bootstrap – 使用Twitter Bootstrap Modal的Ember
- Angular 4 快速入门
- 从Angular 4中的选择选项中获取价值
- 我为什么选择Angular2?
- scala – Pipeline中Spark Dataframe中的OneHotEncoder
- bash中的getopts,脚本之前正在工作,现在我感到很困惑
- angularjs asp.net json datetime parser
- vim /etc/profile 写入时 出现 E121:无法打开并写入文件解决
- bootstrap div 居中
