gpt4 book ai didi

使用泛型和案例类方法的 Scala 特征

转载 作者:行者123 更新时间:2023-12-01 22:33:08 29 4
gpt4 key购买 nike

我有以下情况/代码;

trait Model {
def myField: String
}

case class MyModel(myField: String) extends Model

在为我的模型类创建 DAO 的传统模型中,我想创建一个包含一些通用 CRUD 操作的 DAO 特征。注意...持久性框架在这里无关紧要...问题在于在使用泛型的特征中使用案例类方法。

话虽如此,我想创建以下特征;

trait DAO[M <: Model] {
def insert(model: M): M = {
... do work
m.copy(myField="someval")
}
}

在这种情况下,代码无法编译,因为泛型 M 对成为“案例类”一无所知。如果这里有一些简单的解决方案,是否可以将泛型声明为需要是一种案例?或者 Model 特征是否应该声明任何扩展类都必须实现的复制方法,并通过作为案例类来实现?

最佳答案

If there some easy solution here, can a generic be declared as needing to be a type of case?

没有。

Or should the Model trait declare a copy method that any extending class has to implement and by being a case class it does so?

不幸的是,也没有。一个抽象的 copy 方法不会工作有两个原因。

首先,如果你在你的特征中声明一个抽象的copy方法,它实际上会阻止扩展它的案例类自动生成它,你不得不自己实现它。

其次,很难要求像这样的通用方法。我的意思是,当你声明一个抽象方法时,你需要指定完整的签名。但是,人们会假设所有扩展 Model 的案例类都不会具有相同的签名,因此它们不能具有相同的 copy 方法。它只是不能那样工作。

不幸的是,这让您需要自己为子类实现类似的方法。我使用 F-bounded 多态性来填充 id:

trait Model[M <: Model[M]] { this: M =>
def id: Option[Long]
def withId(id: Long): M
}

case class MyModel(id: Option[Long], ...) extends Model[MyModel] {
def withId(id: Long): MyModel = this.copy(id = Some(id))
}

trait DAO[M <: Model[M]] {
def insert(model: M): M = {
... do work
m.withId(someGeneratedId)
}
}

重复的代码有点丑陋,但足以忍受。也许可以通过反射或宏来完成类似的事情,但这可能并不简单。

关于使用泛型和案例类方法的 Scala 特征,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28597363/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com