Scala:Trait Mixin与抽象基类
我有一个抽象基类(Base),它有一些为它定义的堆栈特性(StackingTrait).
trait Base { def foo } trait StackingTrait extends Base { abstract override def foo { super.foo } } 使用以下语法实现子类非常方便,但是这不起作用,因为编译器说foo需要使用override声明,然后在重新编译时使用抽象覆盖,这是无效的,因为Impl是一个类. class Impl extends Base with StackingTrait { def foo {} } 我想不出为什么不允许这样的语法的一个很好的理由; foo在逻辑上用Impl定义,因此在概念上对堆叠的排序保持不变. 注意: class ImplHelper extends Base { def foo {} } class Impl extends ImplHelper with StackingTrait 为什么没有编译所需的语法,是否有一个优雅的解决方案? 解决方法
我的理解是,虽然错误消息可能令人困惑,但行为是正确的.
foo在StackingTrait中被声明为抽象覆盖,因此在混合StackingTrait的任何具体类中,在StackingTrait之前必须有一个具体的(未标记为抽象的)foo实现(相对于线性化顺序).这是因为super在线性化顺序中引用了之前的特性,因此在混合StackingTrait之前肯定需要具体实现foo,或者super.foo是无意义的. 当你这样做: class Impl extends Base with StackingTrait { def foo {} } 线性化顺序是Base< - StackingTrait< - Impl. StackingTrait之前的唯一特性是Base和Base没有定义foo的具体实现. 但是当你这样做时: traitImplHelper extends Base { def foo {} } class Impl extends ImplHelper with StackingTrait 线性化顺序变为:Base< - ImplHelper< - StackingTrait< - Impl 值得一提的是,如果你在StackingTrait之后混合了ImplHelper(就像在类Impl中使用ImplHelper扩展StackingTrait那样),你将再次遇到同样的问题并且无法编译. 所以,这看起来与我相当一致. trait Base { protected def fooImpl def foo { fooImpl } } trait StackingTrait extends Base { abstract override def foo { super.foo } } class Impl extends Base with StackingTrait { protected def fooImpl {} } 就像在原始版本中一样,你强制每个具体类实现foo(以fooImpl的形式),这次它会编译.这里的缺点是,虽然fooImpl不能调用super.foo(它没有任何意义,并且将进入无限循环),编译器不会警告你. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |