1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
|
trait A {
type m[+x]
}
trait A2 {
type m[+x <: String]
}
trait A3 {
type m[x]
}
trait FooCov[+x]
trait FooCon[-x]
trait FooBound[+x <: String]
trait BOk1 extends A {
type m[+x] = FooCov[x]
}
trait BOk2 extends A2 {
type m[+x <: String] = FooBound[x]
}
trait BOk3 extends A2 {
type m[+x] = FooCov[x] // weaker bound
}
trait BOk4 extends A3 {
type m[+x] = FooCov[x] // weaker variance
}
// there are two aspects to check:
// does type alias signature (not considering RHS) correspond to abstract type member in super class
// does RHS correspond to the type alias sig
trait BInv extends A{
type m[x] = FooCov[x] // error: invariant x in alias def
}
trait BCon extends A{
type m[-x] = FooCon[x] // error: contravariant x
}
trait BBound extends A{
type m[+x <: String] = FooBound[x] // error: x with stricter bound
}
|