假设我们有一个特征,它有一些值和一些操作。
trait Foo {
type Self <: Foo
val x: Int
def withX(x: Int): Self
}
这是使用抽象类型实现的。我们在Self上有一个绑定类型,可以这样实现:
case class Foo1(x: Int) extends Foo {
type Self = Foo1
def withX(x: Int) = copy(x = x)
}
没关系。我们可以使用该方法,我们看到该类型是静态保留的。
scala> Foo1(10).withX(5)
res0: Foo1 = Foo1(5)
当我们想要使用特征类型而不是具体类型的操作时,问题就开始了:
object Foo {
//Error:(13, 43) type mismatch;
//found : f.Self
//required: A
// def setFive[A <: Foo](f: A): A = f.withX(5)
}
好吧,我们不能完全这样做,因为编译器不知道Foo#Self将被分配到什么类型。但我们知道它是同一类型。
当然使用丑陋的方法可以正常工作:
object Foo {
// Ugly type signature
def setFiveValid[A <: Foo](f: A): A#Self = f.withX(5)
// Another ugly type signature
def setFiveValid2[A <: Foo](f: A): f.Self = f.withX(5)
}
他们都没有非常清楚地表达意图。
我们可以使用类型类来解决它。
case class Foo2(x: Int)
trait FooOps[A] extends Any {
def a: A
def withX(x: Int): A
}
object Foo2 {
implicit class Foo2Ops(val a: Foo2) extends AnyVal with FooOps[Foo2] {
def withX(x: Int) = a.copy(x = x)
}
}
object Foo {
// View bounds approach.
def setFiveValid3[A <% FooOps[A]](f: A): A = f.withX(5)
}
然而,这仍然非常嘈杂。
是否有更好的方法来实施setFive
?
修改1
自我类型的主要问题是:
Error:(24, 11) type mismatch;
found : app.models.world.WObject.WorldObjUpdate[self.Self] => app.models.world.WObject.WorldObjUpdate[self.Self]
(which expands to) app.models.game.events.Evented[(app.models.world.World, self.Self)] => app.models.game.events.Evented[(app.models.world.World, self.Self)]
required: app.models.world.WObject.WorldObjUpdate[self.Self] => app.models.game.events.Evented[(app.models.world.World, Self)]
(which expands to) app.models.game.events.Evented[(app.models.world.World, self.Self)] => app.models.game.events.Evented[(app.models.world.World, Self)]
identity
^
然后再次使用奇怪的签名和样板:
def attackReachable(
data: WObject.WorldObjUpdate[Self]
): WObject.WorldObjUpdate[data.value._2.Self]
答案 0 :(得分:5)
你可以走“F-bounded quantification”之路:
trait Foo[F <: Foo[F]] {
def withX(x: Int): F
}
object Foo {
def setFive[F <: Foo[F]](f: F): F = f.withX(5)
}
我成功地使用了这个,但它的代价是不得不在任何地方写F <: Foo[F]]
。
答案 1 :(得分:1)
最佳签名是您建议的路径依赖类型:
// Another ugly type signature
def setFiveValid2[A <: Foo](f: A): f.Self = f.withX(5)
您甚至不需要type参数。将f
键入Foo
即可(除非您在真实情境中需要A
其他内容):
def setFiveValid3(f: Foo): f.Self = f.withX(5)
这并不难看。相反,它是路径依赖类型的完美用途之一。当你说它没有明确表达意图时,我也不同意:你清楚地说明结果将具有你所给出的论证类型Self
。