我有一个Scala案例类,它具有某些val作为参数.我有几种方法可以使用这个案例类.假设我的case类定义如下:
case class CaseA(a: Int, b: List[CaseB])
case class CaseB(a: Int, b: text, c: Boolean)
Run Code Online (Sandbox Code Playgroud)
CaseA和CaseB都代表我的案例类的域模型.但是当我在我的应用程序中使用它们时,我想基于CaseB的布尔字段向CaseA添加一些行为.例如,让我们假设在CaseA中,val b有4个CaseB对象.我想添加一个贯穿b的行为,并告诉我CaseB中的任何一个元素是否将val c设置为true.
我所做的是CaseA的以下实现:
case class CaseA(a: Int, b: List[CaseB], c: Boolean)
Run Code Online (Sandbox Code Playgroud)
但是这种方法的问题在于,当我从数据库加载我的CaseA对象时,我不需要具有val c,因为val c是计算的而不是CaseA对象的一部分.我更进了一步,修改了我的CaseA,如下所示:
case class CaseA(a: Int, b: List[CaseB], c: Option[Boolean])
Run Code Online (Sandbox Code Playgroud)
我可以看到它已经变得丑陋了.如何向我的case类添加行为,以便val c不需要成为CaseA对象的一部分,而是每次在运行时计算它?
如果你不需要所有元素的c,我会使用一个惰性值
case class CaseB(a:Int,b:String,c:Boolean)
case class CaseA(a:Int,b:List[CaseB]){
lazy val c = b exists (_.c)
}
val ca = CaseA(42, List(
CaseB(1,"hello",false),
CaseB(2,",",false),
CaseB(3,"world",true))
)
println(ca.c)
Run Code Online (Sandbox Code Playgroud)
要回答一些评论问题,如果您想使用mixin-trait,您可以执行以下操作:
case class CaseB(a:Int,b:String,c:Boolean)
case class CaseA(a:Int,b:List[CaseB])
trait CanCheck extends CaseA{
lazy val c = b exists (_.c)
}
val ca = new CaseA(42, List(
CaseB(1,"hello",false),
CaseB(2,",",false),
CaseB(3,"world",true))
) with CanCheck
println(ca.c)
Run Code Online (Sandbox Code Playgroud)
请注意,您需要使用new关键字来使其工作