Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Scala: Extending inner class, without reference to outer class

I can extend an inner class/trait inside the outer class or inside a class derived from the outer class. I can extend an inner class of a specific instance of an outer class as in:

class Outer
{
  class Inner{}
}

class OtherCl(val outer1: Outer)
{
  class InnA extends outer1.Inner{}
}

Note: even this seems to compile fine producing very interesting possibilities:

trait OuterA
{ trait InnerA } 

trait OuterB
{ trait InnerB }

class class2(val outerA1: OuterA, val outerB1: OuterB)
{ class Inner2 extends outerA1.InnerA with outerB1.InnerB }

But this won't compile:

class OtherCl extends Outer#Inner

As far as I can see I'm trying to extend a parametrised class where the type parameter is an instance of the outer class so something to the effect of

class OtherCl[T where T is instance of Outer] extends T.Inner

So is the anyway to extend an inner class/ trait that's inside an outer trait/class without reference to the outer trait/class?

I am not looking to instantiate the derived inner class without an instance of the outer class only declare its type.

like image 875
Rich Oliver Avatar asked Dec 15 '22 23:12

Rich Oliver


1 Answers

You can use a trait with a self-type to do something similar. Suppose for example that we have the following:

class Outer(val x: Int) {
  class Inner {
    def y = x
  }
}

And we want to add some functionality to Inner without having an Outer around:

trait MyInner { this: Outer#Inner =>
  def myDoubledY = this.y * 2
}

Now when we instantiate an Inner we can mix in MyInner:

scala> val o = new Outer(21)
o: Outer = Outer@72ee303f

scala> val i = new o.Inner with MyInner
i: o.Inner with MyInner = $anon$1@2c7e9758

scala> i.myDoubledY
res0: Int = 42

It's not exactly what you want, but close.

like image 57
Travis Brown Avatar answered Feb 15 '23 10:02

Travis Brown