我希望能够做到这样的事情:
trait A {
def f(): ???_THE_EXTENDING CLASS
}
class C extends A {
def f() = self
}
class D extends A {
def f() = new D
}
class Z extends D {
def f() = new Z
}
鉴于上面的代码
,以下内容无法编译class Bad1 extends A {
def f() = "unrelated string"
}
class Bad2 extends A {
def f() = new C // this means that you can't just define a type parameter on
// A like A[T <: A] with f() defined as f: T
}
class Bad3 extends D // f() now doesn't return the correct type
这种关系有名字吗?它是如何在Scala中注释/实现的?
以下类型的作品,如您所见:
scala> trait A {
| def f: this.type
| }
defined trait A
scala> class C extends A {
| def f = this
| }
defined class C
scala> class D extends A {
| def f = new D
| }
<console>:7: error: type mismatch;
found : D
required: D.this.type
def f = new D
^
有没有办法解决这个问题?
使用第二个系统,我可以做到这一点,这很好地符合D类的定义:
scala> trait A[T <: A[T]] { def f(): T }
defined trait A
// OR
scala> trait A[T <: A[T]] { self: T =>
| def f(): T
| }
scala> class C extends A[C] { def f() = new C }
defined class C
scala> class D extends C
defined class D
scala> (new D).f
res0: C = C@465fadce
答案 0 :(得分:13)
我担心没有可能知道扩展类的扩展类是什么。
最接近你想要的东西类似于C ++中众所周知的奇怪重复模板模式(CRTP)。
trait A[T <: A[T]] {
def f(): T;
}
class C extends A[C] {
def f() = new C
}
class D extends A[D] {
def f() = new D
}
答案 1 :(得分:6)
您可以做的一件事是返回类型this.type
:
trait A {
def f(): this.type
}
class C extends A {
def f() = this
}
class D extends A {
def f() = this
}
class Z extends D {
override def f() = this
def x = "x"
}
println((new Z).f().x)
这对构建者很有用。
答案 2 :(得分:4)
这是另一种可能的解决方案。它是自我类型+类型参数的组合:
trait A[T <: A[T]] { self: T =>
def f(): T
}
class Z extends A[Z] {
override def f() = new Z
def x = "x"
}
println((new Z).f().x)
在这里您可以找到有关此解决方案的更多信息: