Scala中具有行为的案例类

spa*_*rkr 4 scala class case

我有一个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对象的一部分,而是每次在运行时计算它?

Fel*_*lix 5

如果你不需要所有元素的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关键字来使其工作