Подтвердить что ты не робот

Как вызвать супер метод при переопределении метода по признаку

Похоже, что можно изменить реализацию метода в классе с таким признаком, как:

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
}

Однако ни одна из этих альтернатив не компилируется. Любая идея, как это можно сделать?

4b9b3361

Ответ 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

Я не знаю, можете ли вы внести следующие изменения, но эффект, который вы хотите, может быть достигнут путем введения дополнительного признака (я назову его 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

Ответ 3

abstract override - это механизм, ака стекируемые черты. Стоит добавить, что линеаризация считается, потому что то, что определяет, что означает super.

Этот вопрос является отличным дополнением к каноническому Q & A для самонастройки vs расширения.

Если наследование неоднозначно с само-типами:

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 = [email protected]

scala> .f
res6: String = bar

scala> new Foo with Bar { override def f = super[Foo].f }
res7: Foo with Bar = [email protected]

scala> .f
res8: String = foo

или

scala> new Bar with Foo {}
res9: Bar with Foo = [email protected]

scala> .f
res10: String = foo