I have the following simple program that defines 2 identical upper bounds for type parameter and abstract type alias respectively:
package scala.spike.typeBoundInference
object Example1 {
trait Domain {
}
trait Impl {
type DD <: Domain
type GG <: StaticGraph[DD]
type A1
type A2
type A3
// ... this type list can go very long
// so inlining them as generic type parameters is impossible
final type Builder = StaticGraph.Builder[DD, GG]
}
trait DSL[I <: Impl] {
val impl: StaticGraph.Builder[I#DD, I#GG]
}
trait StaticGraph[T <: Domain] {}
object StaticGraph {
trait Builder[D <: Domain, G <: StaticGraph[D]] {}
}
}
However, scala refuse to compile it:
Error:(16, 27) type arguments [I#DD,I#GG] do not conform to trait Builder's type parameter bounds [D <: scala.spike.typeBoundInference.Example1.Domain,G <: scala.spike.typeBoundInference.Example1.StaticGraph[D]] val impl: StaticGraph.Builder[I#DD, I#GG]
What could possibly go wrong here?
DD <: Domain check
GG <: StaticGraph[DD] check
there is no reason scala think it is unsafe.
In the meantime, I found that if class StaticGraph[T] is declared as covariant scala compiler will run successfully. This is even worse (for some reason StaticGraph[T] has to be invariant), as type bound GG <: StaticGraph[DD] means that if type DD is determined, then GG is a subclass of StaticGraph[DD], but not necessary a subclass of StaticGraph[Domain], which is exactly what I want here.
UPDATE 1: I've read all the answers and comments and somehow got the impression that the core reason is that there is no guarantee that for any instance i
of Impl
, the type bound only guarantee that type
i.DD <:< Impl#DD
and Imp#GG <:< StaticGraph[Impl#DD]
but not StaticGraph[i.DD] <:< StaticGraph[Impl#GG]
thus i.GG <:< StaticGraph[i.DD]
is also not guaranteed.
However, I've done a quick experiment to verify this idea, which turns out to be not ture:
object Example1 {
trait Domain {}
class D1 extends Domain {}
trait Impl {
type DD <: Domain
type GG <: StaticGraph[DD]
}
class StaticGraph[T <: Domain] {}
object Impl1 extends Impl {
type DD = D1
type GG = StaticGraph[Domain]
}
//or this:
val impl = new Impl {
type DD = D1
type GG = StaticGraph[Domain]
}
}
In this case compiler throw an error:
Error:(19, 10) overriding type GG in trait Impl with bounds <: scala.spike.TypeBoundInference.Example1.StaticGraph[scala.spike.TypeBoundInference.Example1.Impl1.DD]; type GG has incompatible type type GG = StaticGraph[Domain]
If you think the type constraint doesn't hold for some instances, could you give me counter example?
UPDATE2: turns out that according to the answer, this is true:
i.GG <:< StaticGraph[i.DD]
but this maybe false:
Impl#GG <:< StaticGraph[Impl#GG]
.
so in the context of DSL this may also be false:
I#GG <:< StaticGraph[I#GG]
(3)
But this is only part of the puzzle, to prove that it is type unsafe, we have to construct a counter example of DSL[I] that invalidates condition (3). So the old question remains: is it possible to construct a counter example?
OK Problem solved:
import scala.language.higherKinds
object Example5 {
trait Domain {}
trait D1 extends Domain
trait Impl {
type DD <: Domain
type GG[T <: Domain] <: StaticGraph[T]
}
trait DSL[I <: Impl] {
val impl: Builder[I#DD, I#GG]
}
trait StaticGraph[T <: Domain] {}
trait Builder[D <: Domain, G[T <: Domain] <: StaticGraph[T]] {}
}
I can't believe I have to use higher kind for such banal matter :-<
Why it compiles? It decouples type constraint and delay it until it becomes necessary. (this is the only explanation I can think of)