- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
Abstract overrides?
Fri, 2009-06-26, 18:46
I'm confused as to the semantics of abstract overrides. From
Programming in Scala, my interpretation was that a trait with a method
defined with the "abstract override" modifier could only be mixed into
a class that provides a concrete implementation of the method.
However, this doesn't appear to quite be the case, at least in 2.7.5:
object Test3 {
type Binding = String => String
trait ABinding extends Binding {
abstract override def apply(s: String): String = super.apply(s) + "a"
}
object BBinding extends Binding with ABinding {
override def apply(s: String) = s + "b"
}
def main(args: Array[String]) {
println(BBinding("main"))
}
}
knuttycombe@knuttycombe-ubuntu:~/tmp/scala$ scalac Test3.scala
Test3.scala:9: error: error overriding method apply in trait ABinding
of type (String)String;
method apply needs `abstract override' modifiers
override def apply(s: String) = s + "b"
^
one error found
Here, I expected that BBinding would provide the concrete
implementation of apply() needed to allow ABinding to be mixed in. Why
is this not the case?
The following works:
object Test3 {
type Binding = String => String
trait ABinding extends Binding {
abstract override def apply(s: String): String = super.apply(s) + "a"
}
private class BBinding extends Binding {
override def apply(s: String) = s + "b"
}
object CBinding extends BBinding with ABinding
def main(args: Array[String]) {
println(CBinding("main"))
}
}
So once there's a concrete class that defines the base case, things
work fine. Is it universally the case that a class cannot directly
extend a trait which defines a method as "abstract override def" by
providing a concrete implementation, and that an intermediary class
must be defined?
Thanks,
Kris
Sat, 2009-06-27, 17:07
#2
Re: Abstract overrides?
trait ABinding should be mixed in, so it really overrides the apply method
object Test3 {
type Binding = String => String
trait ABinding extends Binding {
abstract override def apply(s: String): String = super.apply(s) + "a"
}
class BBinding extends Binding {
def apply(s: String) = s + "b"
}
def main(args: Array[String]) {
val bind = new BBinding with ABinding // <------- here it happens
println(bind("main"))
}
}
Frank
On Fri, Jun 26, 2009 at 2:46 PM, Kris Nuttycombe <kris.nuttycombe@gmail.com> wrote:
Binding = Function1[String,String], so it defines apply(String): String. Here is the order of inheritance: [Function1's supers] Function1 ABinding { override def apply(s : String) = s + "b" } -- a structural type, iirc. I changed the abstract override defintion just to get this compiled. Here is the compiled tree of it: scala> :tree Test3
object Test3 extends scala.ScalaObject {
def <init>() = {
super.<init>();
()
};
type Binding = _root_.scala.Function1[String, String];
abstract trait ABinding extends Binding with scala.ScalaObject {
def $init$() = {
()
};
def x(s: String): String = "a"
};
object BBinding extends Binding with ABinding with scala.ScalaObject {
def <init>() = {
super.<init>();
()
};
override def apply(s: String) = s.$plus("b")
}
} (ModuleDef) See?
--
Daniel C. Sobral
Something I learned in academia: there are three kinds of academic reviews: review by name, review by reference and review by value.