【问题标题】:How to call super method when overriding a method through a trait通过特征覆盖方法时如何调用超级方法
【发布时间】:2013-10-08 00:19:25
【问题描述】:

看起来可以更改具有如下特征的类上的方法的实现:

trait Abstract { self: Result =>
    override def userRepr = "abstract"
}

abstract class Result {
    def userRepr: String = "wtv"
}

case class ValDefResult(name: String) extends Result {
    override def userRepr = name
}

val a = new ValDefResult("asd") with Abstract
a.userRepr

可在此处获取实时代码:http://www.scalakata.com/52534e2fe4b0b1a1c4daa436

但现在我想调用函数的先前或超级实现,如下所示:

trait Abstract { self: Result =>
    override def userRepr = "abstract" + self.userRepr
}

trait Abstract { self: Result =>
    override def userRepr = "abstract" + super.userRepr
}

但是,这些替代方案都无法编译。知道如何实现吗?

【问题讨论】:

标签: scala polymorphism mixins traits


【解决方案1】:

这是我一直在寻找的答案。感谢 Shadowlands 使用 Scala 的 abstract override 功能为我指明了正确的方向。

trait Abstract extends Result {
    abstract override def userRepr = "abstract " + super.userRepr
}

abstract class Result {
    def userRepr: String = "wtv"
}

case class ValDefResult(name: String) extends Result {
    override def userRepr = name
}

val a = new ValDefResult("asd") with Abstract
a.userRepr

可在此处获取实时代码:http://www.scalakata.com/52536cc2e4b0b1a1c4daa4a4

抱歉,示例代码令人困惑,我正在编写一个处理 Scala AST 的库,但没有足够的灵感来更改名称。

【讨论】:

    【解决方案2】:

    我不知道您是否可以进行以下更改,但是可以通过引入额外的 trait(我将其称为 Repr)并在中使用 abstract override 来实现您想要的效果Abstract 特征:

    trait Repr {
        def userRepr: String
    }
    
    abstract class Result extends Repr {
        def userRepr: String = "wtv"
    }
    
    case class ValDefResult(name: String) extends Result {
        override def userRepr = name
    }
    
    trait Abstract extends Repr { self: Result =>
        abstract override def userRepr = "abstract-" + super.userRepr // 'super.' works now
    }
    

    您的示例用法现在给出:

    scala> val a = new ValDefResult("asd") with Abstract
    a: ValDefResult with Abstract = ValDefResult(asd)
    
    scala> a.userRepr
    res3: String = abstract-asd
    

    【讨论】:

    • 虽然,我认为我不需要将 Repr 特征添加到我的特定情况,感谢 Scala 中的抽象覆盖功能为我指明了正确的方向。我发现这篇文章解释了这个功能:artima.com/scalazine/articles/stackable_trait_pattern.html
    • 非常好的解决方案,但提供了一个更低的基本特征,称为 Repr 以包含需要重写的方法,很好玩
    【解决方案3】:

    abstract override 是机制,也就是可堆叠的特征。值得补充的是,线性化很重要,因为这决定了 super 的含义。

    这个问题是the canonical Q&A on self-type vs extension 的一个很好的补充。

    继承与自我类型不明确的地方:

    scala> trait Bar { def f: String = "bar" }
    defined trait Bar
    
    scala> trait Foo { _: Bar => override def f = "foo" }
    defined trait Foo
    
    scala> new Foo with Bar { }
    <console>:44: error: <$anon: Foo with Bar> inherits conflicting members:
      method f in trait Foo of type => String  and
      method f in trait Bar of type => String
    (Note: this can be resolved by declaring an override in <$anon: Foo with Bar>.)
                  new Foo with Bar { }
                      ^
    

    那么显然,你可以选择:

    scala> new Foo with Bar { override def f = super.f }
    res5: Foo with Bar = $anon$1@57a68215
    
    scala> .f
    res6: String = bar
    
    scala> new Foo with Bar { override def f = super[Foo].f }
    res7: Foo with Bar = $anon$1@17c40621
    
    scala> .f
    res8: String = foo
    

    scala> new Bar with Foo {}
    res9: Bar with Foo = $anon$1@374d9299
    
    scala> .f
    res10: String = foo
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2016-07-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2011-12-13
      • 1970-01-01
      • 2015-08-08
      • 1970-01-01
      相关资源
      最近更新 更多