This page is no longer maintained — Please continue to the home page at www.scala-lang.org

abstract override weirdness

9 replies
nilskp
Joined: 2009-01-30,
User offline. Last seen 1 year 27 weeks ago.
Any way to get this to work?

trait SimpleTrait {
  def convert(number: String): Int
}

trait OverridingTrait extends SimpleTrait {
  abstract override def convert(number: String) = try {
    super.convert(number)
  } catch {
    case e: IllegalArgumentException => throw new RuntimeException("Not a number: " + number)
  }
}

class Converter extends SimpleTrait with OverridingTrait {
  def convert(number: String) = Integer.parseInt(number)
}

Jan Lohre
Joined: 2008-12-17,
User offline. Last seen 42 years 45 weeks ago.
Re: abstract override weirdness
Hi Nils,

it is always helpful to provide information on what is not working and what you want to achieve (in order to get alternative proposals on how to achieve it).

Kind regards,
Jan

2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
Any way to get this to work?

trait SimpleTrait {
  def convert(number: String): Int
}

trait OverridingTrait extends SimpleTrait {
  abstract override def convert(number: String) = try {
    super.convert(number)
  } catch {
    case e: IllegalArgumentException => throw new RuntimeException("Not a number: " + number)
  }
}

class Converter extends SimpleTrait with OverridingTrait {
  def convert(number: String) = Integer.parseInt(number)
}


Jan Lohre
Joined: 2008-12-17,
User offline. Last seen 42 years 45 weeks ago.
Re: abstract override weirdness
Having said this I had a look at your example. Be cautious with what I say below because I am missing pratical experience with more then play code in scala.

You are calling the super implementation of convert without actually inheriting a concrete implementation (I guess thats why the abstract modifier is neccessary)
That means there has to be some super-class/trait for your OverridingTrait so that the super call makes sense. You could introduce a ConcreteTrait to give an implementation and do it as in the following:

trait SimpleTrait {
  def convert(number: String): Int
}

trait OverridingTrait extends SimpleTrait {
  abstract override def convert(number: String) = try {
    super.convert(number)
  } catch {
    case e: IllegalArgumentException => throw new RuntimeException("Not a number: " + number)
  }
}

trait ConcreteTrait extends SimpleTrait {
  override def convert(number: String) = Integer.parseInt(number)
}

class Converter extends SimpleTrait with ConcreteTrait with OverridingTrait {
}

object Main {
  def main(args : Array[String]) : Unit = {
    val c = new Converter
    Console.println(c.convert("4a"))
  }
}

As trait linearization will take place from right to left the super call in OverridingTrait will call the concrete implementation in ConcreteTrait.

As I said this is without much practical experience with complex trait/class hierarchies.

Kind regards,
Jan

2009/8/2 Jan Lohre <jan.lohre@googlemail.com>
Hi Nils,

it is always helpful to provide information on what is not working and what you want to achieve (in order to get alternative proposals on how to achieve it).

Kind regards,
Jan

2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
Any way to get this to work?

trait SimpleTrait {
  def convert(number: String): Int
}

trait OverridingTrait extends SimpleTrait {
  abstract override def convert(number: String) = try {
    super.convert(number)
  } catch {
    case e: IllegalArgumentException => throw new RuntimeException("Not a number: " + number)
  }
}

class Converter extends SimpleTrait with OverridingTrait {
  def convert(number: String) = Integer.parseInt(number)
}



nilskp
Joined: 2009-01-30,
User offline. Last seen 1 year 27 weeks ago.
Re: abstract override weirdness
On Sun, Aug 2, 2009 at 6:34 AM, Jan Lohre <jan.lohre@googlemail.com> wrote:
You are calling the super implementation of convert without actually inheriting a concrete implementation (I guess thats why the abstract modifier is neccessary)
Yes.

That means there has to be some super-class/trait for your OverridingTrait so that the super call makes sense.
Yes, that's what the Converter class is.
 
You could introduce a ConcreteTrait to give an implementation and do it as in the following:
 
trait ConcreteTrait extends SimpleTrait {
  override def convert(number: String) = Integer.parseInt(number)
}

class Converter extends SimpleTrait with ConcreteTrait with OverridingTrait {
}

Ok, for some reason that seems to work. Any idea why I have to have the implementation in a trait instead of a class? A bug?
BTW, I' trying this in Eclipse with 2.7.5 and am getting this error:
error overriding method convert in trait OverridingTrait of type (String)Int;
 method convert needs `override' modifier

Randall R Schulz
Joined: 2008-12-16,
User offline. Last seen 1 year 29 weeks ago.
Re: abstract override weirdness

On Sunday August 2 2009, Nils Kilden-Pedersen wrote:
> ...
>
> Ok, for some reason that seems to work. Any idea why I have to have
> the implementation in a trait instead of a class? A bug?
> BTW, I' trying this in Eclipse with 2.7.5 and am getting this error:
> error overriding method convert in trait OverridingTrait of type
> (String)Int;
> method convert needs `override' modifier

If you're inhereting a member with an implementation (not just a
declaration), then you have to explicitly state your intent to override
that inherited method (or val or var) by supplying the "override"
keyword.

Randall Schulz

nilskp
Joined: 2009-01-30,
User offline. Last seen 1 year 27 weeks ago.
Re: abstract override weirdness
On Sun, Aug 2, 2009 at 10:04 AM, Randall R Schulz <rschulz@sonic.net> wrote:
If you're inhereting a member with an implementation (not just a
declaration), then you have to explicitly state your intent to override
that inherited method (or val or var) by supplying the "override"
keyword.

After I sent it, I realized that someone was going to make that comment. I have of course tried that. And it leads to this error:

error overriding method convert in trait OverridingTrait of type (String)Int;
 method convert needs `abstract override' modifiers

And the next error is as expected:

`abstract override' modifier only allowed for members of traits

Ahh, yes, not very helpful.

Jan Lohre
Joined: 2008-12-17,
User offline. Last seen 42 years 45 weeks ago.
Re: abstract override weirdness


2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
On Sun, Aug 2, 2009 at 6:34 AM, Jan Lohre <jan.lohre@googlemail.com> wrote:
You are calling the super implementation of convert without actually inheriting a concrete implementation (I guess thats why the abstract modifier is neccessary)
Yes.

That means there has to be some super-class/trait for your OverridingTrait so that the super call makes sense.
Yes, that's what the Converter class is.

No, the Converter class is a subclass/implementation of the traits, not a super-class.
 

 
You could introduce a ConcreteTrait to give an implementation and do it as in the following:
 
trait ConcreteTrait extends SimpleTrait {
  override def convert(number: String) = Integer.parseInt(number)
}

class Converter extends SimpleTrait with ConcreteTrait with OverridingTrait {
}

Ok, for some reason that seems to work. Any idea why I have to have the implementation in a trait instead of a class? A bug?
BTW, I' trying this in Eclipse with 2.7.5 and am getting this error:
error overriding method convert in trait OverridingTrait of type (String)Int;
 method convert needs `override' modifier


nilskp
Joined: 2009-01-30,
User offline. Last seen 1 year 27 weeks ago.
Re: abstract override weirdness
On Sun, Aug 2, 2009 at 2:56 PM, Jan Lohre <jan.lohre@googlemail.com> wrote:


2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
On Sun, Aug 2, 2009 at 6:34 AM, Jan Lohre <jan.lohre@googlemail.com> wrote:
You are calling the super implementation of convert without actually inheriting a concrete implementation (I guess thats why the abstract modifier is neccessary)
Yes.

That means there has to be some super-class/trait for your OverridingTrait so that the super call makes sense.
Yes, that's what the Converter class is.

No, the Converter class is a subclass/implementation of the traits, not a super-class.

But why does this work then?:

class Converter extends SimpleTrait {
  def convert(n: String) = Integer.parseInt(n)
}

val c = new Converter with OverridingTrait

So I can use OverridingTrait in an anonymous type definition, but not in a named one. That's gotta be a bug, unless I misunderstand something.
Jan Lohre
Joined: 2008-12-17,
User offline. Last seen 42 years 45 weeks ago.
Re: abstract override weirdness
I will basically let this question for others to answer.

My guess would be that in this case you create kind of an anonymous class (Converter with OverridingTrait) that extends Converter.

But beside that this is only a guess - maybe you are right and it is a bug - and I am not sure I dived deep enough into scala yet to give a qualified anwer; why do you want such a construct in the first place?

2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
On Sun, Aug 2, 2009 at 2:56 PM, Jan Lohre <jan.lohre@googlemail.com> wrote:


2009/8/2 Nils Kilden-Pedersen <nilskp@gmail.com>
On Sun, Aug 2, 2009 at 6:34 AM, Jan Lohre <jan.lohre@googlemail.com> wrote:
You are calling the super implementation of convert without actually inheriting a concrete implementation (I guess thats why the abstract modifier is neccessary)
Yes.

That means there has to be some super-class/trait for your OverridingTrait so that the super call makes sense.
Yes, that's what the Converter class is.

No, the Converter class is a subclass/implementation of the traits, not a super-class.

But why does this work then?:

class Converter extends SimpleTrait {
  def convert(n: String) = Integer.parseInt(n)
}

val c = new Converter with OverridingTrait

So I can use OverridingTrait in an anonymous type definition, but not in a named one. That's gotta be a bug, unless I misunderstand something.

nilskp
Joined: 2009-01-30,
User offline. Last seen 1 year 27 weeks ago.
Re: abstract override weirdness
On Sun, Aug 2, 2009 at 4:48 PM, Jan Lohre <jan.lohre@googlemail.com> wrote:
why do you want such a construct in the first place?

Good question. Maybe I don't. I guess I could just add that functionality in the Converter's implementation. It would just look cleaner the other way, with better separation of concerns. But maybe that's why it's not a bug. Because I don't really need it. 

Copyright © 2012 École Polytechnique Fédérale de Lausanne (EPFL), Lausanne, Switzerland