how to force method to be implemented in concrete subclass from trait

1.4k views Asked by At

I have a method in my trait

def controller: AnyRef

but my concrete class was not implementing that method and it was still compiling. The compiler doesn't let me add abstract to that method either. How can I create a method in a trait that forces it's implementer to implement it?

thanks, Dean

3

There are 3 answers

3
Gabriele Petronella On BEST ANSWER

The compiler enforces that concrete classes implement all the abstracts methods they inherit from superclasses and traits.

If your class was compiling it meant it wasn't concrete, i.e. it was a trait or an abstract class, and you can't force neither to implement the abstract method.

Of course, as soon as you try to obtain a concrete instance the compiler will raise an error as the method is not implemented.

Practical example in the REPL

scala> trait A { def controller: AnyRef }
defined trait A

scala> trait B extends A
defined trait B

scala> abstract class C extends A
defined class C

scala> class D extends A
<console>:8: error: class D needs to be abstract, since method controller in trait A of type => AnyRef is not defined
       class D extends A

scala> new B { }
<console>:10: error: object creation impossible, since method controller in trait A of type => AnyRef is not defined
              new B { }
                  ^

scala> new C
<console>:10: error: class C is abstract; cannot be instantiated
              new C
1
user3366706 On

If you are not getting any compilation error for such case, I think your IDE has some problem, try refresh/clean your project.

If your trait has implementation of "controller" function, then the concrete class(non abstract class) extending trait need not have any implementation for "controller" function. That's the beauty of trait in Scala, not forcing all the concrete class to implement some common methods with same implementation.

0
Dean Hiller On

well, there seems to be some weird scala issue occuring.

When you have a concrete subclass that has a

val controller = new Controller

it seems to affect the traits controller and having your trait call controller results in null being returned. Everything compiles just fine (though I think that might be a scala bug as I don't 'think' that should actually compile).

If it should compile, then can someone please explain why it compiles.